Encrypted messaging app
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

164 lines
4.7 KiB

  1. import 'dart:convert';
  2. import 'package:flutter/material.dart';
  3. import 'package:http/http.dart' as http;
  4. import 'package:pointycastle/export.dart';
  5. import 'package:sqflite/sqflite.dart';
  6. import '/components/flash_message.dart';
  7. import '/models/conversation_users.dart';
  8. import '/models/conversations.dart';
  9. import '/models/my_profile.dart';
  10. import '/utils/encryption/aes_helper.dart';
  11. import '/utils/storage/database.dart';
  12. import '/utils/storage/session_cookie.dart';
  13. Future<void> updateConversation(Conversation conversation, { includeUsers = true } ) async {
  14. String sessionCookie = await getSessionCookie();
  15. Map<String, dynamic> conversationJson = await conversation.payloadJson(includeUsers: includeUsers);
  16. var x = await http.put(
  17. await MyProfile.getServerUrl('api/v1/auth/conversations'),
  18. headers: <String, String>{
  19. 'Content-Type': 'application/json; charset=UTF-8',
  20. 'cookie': sessionCookie,
  21. },
  22. body: jsonEncode(conversationJson),
  23. );
  24. // TODO: Handle errors here
  25. print(x.statusCode);
  26. }
  27. // TODO: Refactor this function
  28. Future<void> updateConversations() async {
  29. RSAPrivateKey privKey = await MyProfile.getPrivateKey();
  30. // try {
  31. var resp = await http.get(
  32. await MyProfile.getServerUrl('api/v1/auth/conversations'),
  33. headers: {
  34. 'cookie': await getSessionCookie(),
  35. }
  36. );
  37. if (resp.statusCode != 200) {
  38. throw Exception(resp.body);
  39. }
  40. List<Conversation> conversations = [];
  41. List<String> conversationsDetailIds = [];
  42. List<dynamic> conversationsJson = jsonDecode(resp.body);
  43. if (conversationsJson.isEmpty) {
  44. return;
  45. }
  46. for (var i = 0; i < conversationsJson.length; i++) {
  47. Conversation conversation = Conversation.fromJson(
  48. conversationsJson[i] as Map<String, dynamic>,
  49. privKey,
  50. );
  51. conversations.add(conversation);
  52. conversationsDetailIds.add(conversation.id);
  53. }
  54. Map<String, String> params = {};
  55. params['conversation_detail_ids'] = conversationsDetailIds.join(',');
  56. var uri = await MyProfile.getServerUrl('api/v1/auth/conversation_details');
  57. uri = uri.replace(queryParameters: params);
  58. resp = await http.get(
  59. uri,
  60. headers: {
  61. 'cookie': await getSessionCookie(),
  62. }
  63. );
  64. if (resp.statusCode != 200) {
  65. throw Exception(resp.body);
  66. }
  67. final db = await getDatabaseConnection();
  68. List<dynamic> conversationsDetailsJson = jsonDecode(resp.body);
  69. for (var i = 0; i < conversationsDetailsJson.length; i++) {
  70. var conversationDetailJson = conversationsDetailsJson[i] as Map<String, dynamic>;
  71. var conversation = findConversationByDetailId(conversations, conversationDetailJson['id']);
  72. conversation.twoUser = AesHelper.aesDecrypt(
  73. base64.decode(conversation.symmetricKey),
  74. base64.decode(conversationDetailJson['two_user']),
  75. ) == 'true';
  76. if (conversation.twoUser) {
  77. MyProfile profile = await MyProfile.getProfile();
  78. final db = await getDatabaseConnection();
  79. List<Map<String, dynamic>> maps = await db.query(
  80. 'conversation_users',
  81. where: 'conversation_id = ? AND user_id != ?',
  82. whereArgs: [ conversation.id, profile.id ],
  83. );
  84. if (maps.length != 1) {
  85. throw ArgumentError('Invalid user id');
  86. }
  87. conversation.name = maps[0]['username'];
  88. } else {
  89. conversation.name = AesHelper.aesDecrypt(
  90. base64.decode(conversation.symmetricKey),
  91. base64.decode(conversationDetailJson['name']),
  92. );
  93. }
  94. await db.insert(
  95. 'conversations',
  96. conversation.toMap(),
  97. conflictAlgorithm: ConflictAlgorithm.replace,
  98. );
  99. List<dynamic> usersData = conversationDetailJson['users'];
  100. for (var i = 0; i < usersData.length; i++) {
  101. ConversationUser conversationUser = ConversationUser.fromJson(
  102. usersData[i] as Map<String, dynamic>,
  103. base64.decode(conversation.symmetricKey),
  104. );
  105. await db.insert(
  106. 'conversation_users',
  107. conversationUser.toMap(),
  108. conflictAlgorithm: ConflictAlgorithm.replace,
  109. );
  110. }
  111. }
  112. // } catch (SocketException) {
  113. // return;
  114. // }
  115. }
  116. Future<void> uploadConversation(Conversation conversation, BuildContext context) async {
  117. String sessionCookie = await getSessionCookie();
  118. Map<String, dynamic> conversationJson = await conversation.payloadJson();
  119. var resp = await http.post(
  120. await MyProfile.getServerUrl('api/v1/auth/conversations'),
  121. headers: <String, String>{
  122. 'Content-Type': 'application/json; charset=UTF-8',
  123. 'cookie': sessionCookie,
  124. },
  125. body: jsonEncode(conversationJson),
  126. );
  127. if (resp.statusCode != 200) {
  128. showMessage('Failed to create conversation', context);
  129. }
  130. }