Merge branch 'lists-enhance-fix-send-status' into 'master'
Lists enhance fix send status See merge request famedly/famedlysdk!15
This commit is contained in:
commit
e652e79a98
|
@ -165,45 +165,65 @@ class Room {
|
||||||
type: "PUT",
|
type: "PUT",
|
||||||
action: "/client/r0/rooms/${id}/send/m.room.message/$txid",
|
action: "/client/r0/rooms/${id}/send/m.room.message/$txid",
|
||||||
data: {"msgtype": "m.text", "body": message});
|
data: {"msgtype": "m.text", "body": message});
|
||||||
if (res["errcode"] == "M_LIMIT_EXCEEDED")
|
if (res is ErrorResponse) client.connection.onError.add(res);
|
||||||
client.connection.onError.add(res["error"]);
|
|
||||||
return res;
|
return res;
|
||||||
}
|
}
|
||||||
|
|
||||||
Future<String> sendTextEvent(String message) async {
|
Future<String> sendTextEvent(String message, {String txid = null}) async {
|
||||||
final String type = "m.room.message";
|
final String type = "m.room.message";
|
||||||
|
String messageID;
|
||||||
final int now = DateTime.now().millisecondsSinceEpoch;
|
final int now = DateTime.now().millisecondsSinceEpoch;
|
||||||
final String messageID = "msg$now";
|
if (txid == null) {
|
||||||
|
messageID = "msg$now";
|
||||||
|
} else
|
||||||
|
messageID = txid;
|
||||||
|
|
||||||
|
// Display a *sending* event and store it.
|
||||||
EventUpdate eventUpdate =
|
EventUpdate eventUpdate =
|
||||||
EventUpdate(type: "timeline", roomID: id, eventType: type, content: {
|
EventUpdate(type: "timeline", roomID: id, eventType: type, content: {
|
||||||
"type": type,
|
"type": type,
|
||||||
"id": messageID,
|
"id": null,
|
||||||
"sender": client.userID,
|
"sender": client.userID,
|
||||||
"status": 0,
|
"status": 0,
|
||||||
"origin_server_ts": now,
|
"origin_server_ts": now,
|
||||||
"content": {
|
"content": {
|
||||||
"msgtype": "m.text",
|
"msgtype": "m.text",
|
||||||
"body": message,
|
"body": message,
|
||||||
|
"txid": messageID,
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
client.connection.onEvent.add(eventUpdate);
|
client.connection.onEvent.add(eventUpdate);
|
||||||
await client.store.transaction(() {
|
await client.store?.transaction(() {
|
||||||
client.store.storeEventUpdate(eventUpdate);
|
client.store.storeEventUpdate(eventUpdate);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
// Send the text and on success, store and display a *sent* event.
|
||||||
final dynamic res = await sendText(message, txid: messageID);
|
final dynamic res = await sendText(message, txid: messageID);
|
||||||
|
|
||||||
if (res is ErrorResponse) {
|
if (res is ErrorResponse) {
|
||||||
client.store.db
|
// On error, set status to -1
|
||||||
.rawUpdate("UPDATE Events SET status=-1 WHERE id=?", [messageID]);
|
eventUpdate.content["status"] = -1;
|
||||||
|
client.connection.onEvent.add(eventUpdate);
|
||||||
|
client.store?.db
|
||||||
|
?.rawUpdate("UPDATE Events SET status=-1 WHERE id=?", [messageID]);
|
||||||
} else {
|
} else {
|
||||||
final String newEventID = res["event_id"];
|
final String newEventID = res["event_id"];
|
||||||
final List<Map<String, dynamic>> event = await client.store.db
|
eventUpdate.content["status"] = 1;
|
||||||
.rawQuery("SELECT * FROM Events WHERE id=?", [newEventID]);
|
eventUpdate.content["id"] = newEventID;
|
||||||
if (event.length > 0) {
|
client.connection.onEvent.add(eventUpdate);
|
||||||
client.store.db.rawDelete("DELETE FROM Events WHERE id=?", [messageID]);
|
|
||||||
} else {
|
// Store the result in database
|
||||||
client.store.db.rawUpdate("UPDATE Events SET id=?, status=1 WHERE id=?",
|
if (client.store != null) {
|
||||||
[newEventID, messageID]);
|
final List<Map<String, dynamic>> eventQuery = await client.store.db
|
||||||
|
.rawQuery("SELECT * FROM Events WHERE id=?", [newEventID]);
|
||||||
|
if (eventQuery.length > 0) {
|
||||||
|
client.store.db
|
||||||
|
.rawDelete("DELETE FROM Events WHERE id=?", [messageID]);
|
||||||
|
} else {
|
||||||
|
client.store.db.rawUpdate(
|
||||||
|
"UPDATE Events SET id=?, status=1 WHERE id=?",
|
||||||
|
[newEventID, messageID]);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
return newEventID;
|
return newEventID;
|
||||||
}
|
}
|
||||||
|
@ -310,6 +330,7 @@ class Room {
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Sets this room as a direct chat for this user.
|
||||||
Future<dynamic> addToDirectChat(String userID) async {
|
Future<dynamic> addToDirectChat(String userID) async {
|
||||||
Map<String, List<String>> directChats =
|
Map<String, List<String>> directChats =
|
||||||
await client.store.getAccountDataDirectChats();
|
await client.store.getAccountDataDirectChats();
|
||||||
|
@ -327,6 +348,7 @@ class Room {
|
||||||
return resp;
|
return resp;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Sends *m.fully_read* and *m.read* for the given event ID.
|
||||||
Future<dynamic> sendReadReceipt(String eventID) async {
|
Future<dynamic> sendReadReceipt(String eventID) async {
|
||||||
final dynamic resp = client.connection.jsonRequest(
|
final dynamic resp = client.connection.jsonRequest(
|
||||||
type: "POST",
|
type: "POST",
|
||||||
|
@ -399,6 +421,7 @@ class Room {
|
||||||
return room;
|
return room;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Creates a timeline from the store. Returns a [Timeline] object.
|
||||||
Future<Timeline> getTimeline(
|
Future<Timeline> getTimeline(
|
||||||
{onTimelineUpdateCallback onUpdate,
|
{onTimelineUpdateCallback onUpdate,
|
||||||
onTimelineInsertCallback onInsert}) async {
|
onTimelineInsertCallback onInsert}) async {
|
||||||
|
|
|
@ -37,6 +37,8 @@ class Timeline {
|
||||||
final onTimelineUpdateCallback onUpdate;
|
final onTimelineUpdateCallback onUpdate;
|
||||||
final onTimelineInsertCallback onInsert;
|
final onTimelineInsertCallback onInsert;
|
||||||
|
|
||||||
|
Set<String> waitToReplace = {};
|
||||||
|
|
||||||
StreamSubscription<EventUpdate> sub;
|
StreamSubscription<EventUpdate> sub;
|
||||||
|
|
||||||
Timeline({this.room, this.events, this.onUpdate, this.onInsert}) {
|
Timeline({this.room, this.events, this.onUpdate, this.onInsert}) {
|
||||||
|
@ -47,20 +49,40 @@ class Timeline {
|
||||||
try {
|
try {
|
||||||
if (eventUpdate.roomID != room.id) return;
|
if (eventUpdate.roomID != room.id) return;
|
||||||
if (eventUpdate.type == "timeline" || eventUpdate.type == "history") {
|
if (eventUpdate.type == "timeline" || eventUpdate.type == "history") {
|
||||||
if (!eventUpdate.content.containsKey("id"))
|
// Is this event already in the timeline?
|
||||||
eventUpdate.content["id"] = eventUpdate.content["event_id"];
|
if (eventUpdate.content["status"] == 1 ||
|
||||||
|
eventUpdate.content["status"] == -1 ||
|
||||||
|
waitToReplace.contains(eventUpdate.content["id"])) {
|
||||||
|
int i;
|
||||||
|
for (i = 0; i < events.length; i++) {
|
||||||
|
if (events[i].content.containsKey("txid") &&
|
||||||
|
events[i].content["txid"] ==
|
||||||
|
eventUpdate.content["content"]["txid"] ||
|
||||||
|
events[i].id == eventUpdate.content["id"]) break;
|
||||||
|
}
|
||||||
|
if (i < events.length) {
|
||||||
|
events[i] = Event.fromJson(eventUpdate.content, room);
|
||||||
|
if (eventUpdate.content["content"]["txid"] is String)
|
||||||
|
waitToReplace.add(eventUpdate.content["id"]);
|
||||||
|
else
|
||||||
|
waitToReplace.remove(eventUpdate.content["id"]);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if (!eventUpdate.content.containsKey("id"))
|
||||||
|
eventUpdate.content["id"] = eventUpdate.content["event_id"];
|
||||||
|
|
||||||
User user = await room.client.store
|
User user = await room.client.store
|
||||||
?.getUser(matrixID: eventUpdate.content["sender"], room: room);
|
?.getUser(matrixID: eventUpdate.content["sender"], room: room);
|
||||||
if (user != null) {
|
if (user != null) {
|
||||||
eventUpdate.content["displayname"] = user.displayName;
|
eventUpdate.content["displayname"] = user.displayName;
|
||||||
eventUpdate.content["avatar_url"] = user.avatarUrl.mxc;
|
eventUpdate.content["avatar_url"] = user.avatarUrl.mxc;
|
||||||
|
}
|
||||||
|
|
||||||
|
Event newEvent = Event.fromJson(eventUpdate.content, room);
|
||||||
|
|
||||||
|
events.insert(0, newEvent);
|
||||||
|
if (onInsert != null) onInsert(0);
|
||||||
}
|
}
|
||||||
|
|
||||||
Event newEvent = Event.fromJson(eventUpdate.content, room);
|
|
||||||
|
|
||||||
events.insert(0, newEvent);
|
|
||||||
if (onInsert != null) onInsert(0);
|
|
||||||
}
|
}
|
||||||
sortAndUpdate();
|
sortAndUpdate();
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
|
|
|
@ -376,7 +376,12 @@ class FakeMatrixApi extends MockClient {
|
||||||
},
|
},
|
||||||
"/client/r0/rooms/!localpart:server.abc/read_markers": (var reqI) => {},
|
"/client/r0/rooms/!localpart:server.abc/read_markers": (var reqI) => {},
|
||||||
},
|
},
|
||||||
"PUT": {},
|
"PUT": {
|
||||||
|
"/client/r0/rooms/!1234:example.com/send/m.room.message/1234":
|
||||||
|
(var reqI) => {
|
||||||
|
"event_id": "42",
|
||||||
|
},
|
||||||
|
},
|
||||||
"DELETE": {
|
"DELETE": {
|
||||||
"/unknown/token": (var req) => {"errcode": "M_UNKNOWN_TOKEN"},
|
"/unknown/token": (var req) => {"errcode": "M_UNKNOWN_TOKEN"},
|
||||||
},
|
},
|
||||||
|
|
|
@ -27,6 +27,7 @@ import 'package:famedlysdk/src/Room.dart';
|
||||||
import 'package:famedlysdk/src/Timeline.dart';
|
import 'package:famedlysdk/src/Timeline.dart';
|
||||||
import 'package:famedlysdk/src/sync/EventUpdate.dart';
|
import 'package:famedlysdk/src/sync/EventUpdate.dart';
|
||||||
import 'package:famedlysdk/src/utils/ChatTime.dart';
|
import 'package:famedlysdk/src/utils/ChatTime.dart';
|
||||||
|
import 'FakeMatrixApi.dart';
|
||||||
|
|
||||||
void main() {
|
void main() {
|
||||||
/// All Tests related to the MxContent
|
/// All Tests related to the MxContent
|
||||||
|
@ -36,21 +37,22 @@ void main() {
|
||||||
int updateCount = 0;
|
int updateCount = 0;
|
||||||
List<int> insertList = [];
|
List<int> insertList = [];
|
||||||
|
|
||||||
|
Client client = Client("testclient", debug: true);
|
||||||
|
client.connection.httpClient = FakeMatrixApi();
|
||||||
|
client.homeserver = "https://fakeServer.notExisting";
|
||||||
|
|
||||||
|
Room room = Room(id: roomID, client: client);
|
||||||
|
Timeline timeline = Timeline(
|
||||||
|
room: room,
|
||||||
|
events: [],
|
||||||
|
onUpdate: () {
|
||||||
|
updateCount++;
|
||||||
|
},
|
||||||
|
onInsert: (int insertID) {
|
||||||
|
insertList.add(insertID);
|
||||||
|
});
|
||||||
|
|
||||||
test("Create", () async {
|
test("Create", () async {
|
||||||
Client client = Client("testclient");
|
|
||||||
client.homeserver = "https://testserver.abc";
|
|
||||||
|
|
||||||
Room room = Room(id: roomID, client: client);
|
|
||||||
Timeline timeline = Timeline(
|
|
||||||
room: room,
|
|
||||||
events: [],
|
|
||||||
onUpdate: () {
|
|
||||||
updateCount++;
|
|
||||||
},
|
|
||||||
onInsert: (int insertID) {
|
|
||||||
insertList.add(insertID);
|
|
||||||
});
|
|
||||||
|
|
||||||
client.connection.onEvent.add(EventUpdate(
|
client.connection.onEvent.add(EventUpdate(
|
||||||
type: "timeline",
|
type: "timeline",
|
||||||
roomID: roomID,
|
roomID: roomID,
|
||||||
|
@ -91,5 +93,48 @@ void main() {
|
||||||
expect(timeline.events[0].getBody(), "Testcase");
|
expect(timeline.events[0].getBody(), "Testcase");
|
||||||
expect(timeline.events[0].time > timeline.events[1].time, true);
|
expect(timeline.events[0].time > timeline.events[1].time, true);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
test("Send message", () async {
|
||||||
|
room.sendTextEvent("test", txid: "1234");
|
||||||
|
|
||||||
|
await new Future.delayed(new Duration(milliseconds: 50));
|
||||||
|
|
||||||
|
expect(updateCount, 4);
|
||||||
|
expect(insertList, [0, 0, 0]);
|
||||||
|
expect(timeline.events[0].content["txid"], "1234");
|
||||||
|
expect(timeline.events[0].id, "42");
|
||||||
|
expect(timeline.events[0].status, 1);
|
||||||
|
|
||||||
|
client.connection.onEvent.add(EventUpdate(
|
||||||
|
type: "timeline",
|
||||||
|
roomID: roomID,
|
||||||
|
eventType: "m.room.message",
|
||||||
|
content: {
|
||||||
|
"type": "m.room.message",
|
||||||
|
"content": {"msgtype": "m.text", "body": "test"},
|
||||||
|
"sender": "@alice:example.com",
|
||||||
|
"status": 2,
|
||||||
|
"id": "42",
|
||||||
|
"origin_server_ts": DateTime.now().millisecondsSinceEpoch
|
||||||
|
}));
|
||||||
|
|
||||||
|
await new Future.delayed(new Duration(milliseconds: 50));
|
||||||
|
|
||||||
|
expect(updateCount, 5);
|
||||||
|
expect(insertList, [0, 0, 0]);
|
||||||
|
expect(timeline.events[0].id, "42");
|
||||||
|
expect(timeline.events[0].status, 2);
|
||||||
|
});
|
||||||
|
|
||||||
|
test("Send message with error", () async {
|
||||||
|
room.sendTextEvent("test", txid: "errortxid");
|
||||||
|
|
||||||
|
await new Future.delayed(new Duration(milliseconds: 50));
|
||||||
|
|
||||||
|
expect(updateCount, 7);
|
||||||
|
expect(insertList, [0, 0, 0, 0]);
|
||||||
|
expect(timeline.events[0].content["txid"], "errortxid");
|
||||||
|
expect(timeline.events[0].status, -1);
|
||||||
|
});
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in a new issue