Hibok
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.
 
 
 
 
 
 

405 lines
11 KiB

  1. import 'dart:convert';
  2. import 'package:chat/data/UserData.dart';
  3. import 'package:chat/generated/i18n.dart';
  4. import 'package:chat/models/ChatMsg.dart';
  5. import 'package:chat/models/group_info_model.dart';
  6. import 'package:chat/models/group_info_server_model.dart';
  7. import 'package:chat/proto/all.pbserver.dart';
  8. import 'package:chat/data/constants.dart';
  9. import 'package:chat/utils/file_cache_mgr.dart';
  10. import 'package:convert/convert.dart';
  11. import 'package:dio/dio.dart';
  12. import 'package:crypto/crypto.dart';
  13. import 'dart:async';
  14. import 'package:http_parser/http_parser.dart';
  15. import 'package:chat/utils/MessageMgr.dart';
  16. import 'package:oktoast/oktoast.dart';
  17. import 'package:chat/utils/msgHandler.dart';
  18. import 'package:chat/utils/sql_util.dart';
  19. var signKey = 'O@MlSxWaeAlc5CYu';
  20. class UploadUtil {
  21. // 工厂模式
  22. factory UploadUtil() => _getInstance();
  23. static UploadUtil get instance => _getInstance();
  24. static UploadUtil _instance;
  25. UploadUtil._internal() {
  26. // 初始化
  27. }
  28. static UploadUtil _getInstance() {
  29. if (_instance == null) {
  30. _instance = new UploadUtil._internal();
  31. }
  32. return _instance;
  33. }
  34. Dio dio;
  35. //进度流
  36. Map<String, StreamController<double>> streamMap = {};
  37. //缓存进度,方便快速显示
  38. Map<String, double> streamLastPercentMap = {};
  39. //上传地址
  40. String uploadUrl;
  41. String rootUrl;
  42. void setUploadUrl(url) {
  43. rootUrl = 'http://' + url;
  44. uploadUrl = 'http://' + url + '/chatresfile/';
  45. BaseOptions options = BaseOptions(
  46. baseUrl: uploadUrl,
  47. headers: {},
  48. );
  49. print(uploadUrl);
  50. dio = new Dio(options);
  51. }
  52. // md5 加密
  53. String generateMd5(String data) {
  54. var content = new Utf8Encoder().convert(data);
  55. var digest = md5.convert(content);
  56. // 这里其实就是 digest.toString()
  57. return hex.encode(digest.bytes);
  58. }
  59. String getSign(Map parameter) {
  60. String k = '';
  61. var keyList = [];
  62. for (var key in parameter.keys) {
  63. keyList.add(key);
  64. }
  65. keyList.sort();
  66. for (var key in keyList) {
  67. k += '$key=${parameter[key]}&';
  68. }
  69. k += 'key=O@MlSxWaeAlc5CYu';
  70. return generateMd5(k);
  71. }
  72. String getFileType(String suffix) {
  73. if (suffix == 'png' || suffix == 'jpg' || suffix == 'jpeg') {
  74. return 'image';
  75. } else if (suffix == 'wav') {
  76. return 'audio';
  77. } else {
  78. return 'video';
  79. }
  80. }
  81. bool isImage(String suffix) {
  82. suffix = suffix.toLowerCase();
  83. return suffix == 'png' || suffix == 'jpg' || suffix == 'jpeg';
  84. }
  85. StreamController<double> getStream(String fileId) {
  86. if (streamMap[fileId] == null) {
  87. streamMap[fileId] = StreamController<double>.broadcast();
  88. }
  89. return streamMap[fileId];
  90. }
  91. Future uploadFile(MsgModel msg, {CancelToken cancelToken}) async {
  92. Response response;
  93. var data = {
  94. "upuserid": msg.from,
  95. 'channeltype': msg.channelType,
  96. "targetid": msg.sessionId,
  97. 'fileid': msg.extraFile,
  98. };
  99. print('上传参数$data');
  100. var path = msg.localFile;
  101. var name = path.substring(path.lastIndexOf("/") + 1, path.length);
  102. String suffix = name.substring(name.lastIndexOf(".") + 1, name.length);
  103. var fileType = getFileType(suffix);
  104. print('文件类型 : $fileType $suffix');
  105. data['sign'] = getSign(data);
  106. data['uploadfile'] = await MultipartFile.fromFile(path,
  107. filename: name, contentType: MediaType(fileType, suffix));
  108. FormData formData = new FormData.fromMap(data);
  109. try {
  110. print('开始上传文件${msg.extraFile} url:${uploadUrl + 'uploadchatfile'}');
  111. var streamController = getStream(msg.extraFile);
  112. response = await dio.post(uploadUrl + 'uploadchatfile', data: formData,
  113. onSendProgress: (int progress, int total) {
  114. streamController.sink.add(progress / total);
  115. streamLastPercentMap[msg.extraFile] = progress / total;
  116. }, cancelToken: cancelToken);
  117. streamController.close();
  118. streamMap.remove(msg.extraFile);
  119. print('uploadFile response $response');
  120. print('###msgContent ${msg.msgContent.length}');
  121. if (response.data['code'] == 0) {
  122. var data = response.data['data'];
  123. print('uploadFile 上传成功');
  124. msg.state = MsgState.Uploaded;
  125. MsgHandler.sendChatMsg(msg);
  126. MessageMgr().emit('Update LastMsg', msg.sessionId);
  127. return data;
  128. } else {
  129. print('uploadFile 上传失败');
  130. msg.state = MsgState.UploadFailed;
  131. MessageMgr().emit('Update LastMsg', msg.sessionId);
  132. return null;
  133. }
  134. } on DioError catch (e) {
  135. if (CancelToken.isCancel(e)) {
  136. print('post请求取消! ' + e.message);
  137. }
  138. showToast('上传文件失败');
  139. msg.state = MsgState.UploadFailed;
  140. print('post请求发生错误:$e');
  141. }
  142. MessageMgr().emit('Update LastMsg', msg.sessionId);
  143. return null;
  144. }
  145. void cancelRequests(CancelToken cancelToken) {
  146. print('取消请求$cancelToken');
  147. cancelToken.cancel("cancelled");
  148. }
  149. String getFullUrl(String imgData, int sessionid, int channelType) {
  150. var data = {
  151. "fileid": imgData,
  152. 'targetid': sessionid,
  153. "userid": UserData().basicInfo.userId,
  154. 'channeltype': channelType
  155. };
  156. var sign = getSign(data);
  157. return uploadUrl +
  158. 'downloadchatfile?targetid=$sessionid&channeltype=$channelType&userid=${UserData().basicInfo.userId}&fileid=$imgData&sign=$sign';
  159. }
  160. Future downloadFile(MsgModel msg, {CancelToken cancelToken}) async {
  161. Response response;
  162. var fullUrl = getFullUrl(msg.extraFile, msg.sessionId, msg.channelType);
  163. print('下载文件$fullUrl');
  164. var path = await FileCacheMgr().genFilePath(msg.extraFile);
  165. if (msg.msgType == ChatType.ShortVoiceChatType.value) {
  166. path = path + '.wav';
  167. } else if (msg.msgType == ChatType.FileChatType.value) {
  168. var fileMsg = FileChat.fromBuffer(msg.msgContent);
  169. var fileName = fileMsg.name;
  170. path = await FileCacheMgr().genFilePath(fileName);
  171. }
  172. print('文件保存路径$path');
  173. try {
  174. var streamController = getStream(msg.extraFile);
  175. response = await dio.download(fullUrl, path,
  176. onReceiveProgress: (int progress, int total) {
  177. streamController.sink.add(progress / total);
  178. streamLastPercentMap[msg.extraFile] = progress / total;
  179. }, cancelToken: cancelToken);
  180. streamController.close();
  181. streamMap.remove(msg.extraFile);
  182. if (response.statusCode == 200) {
  183. msg.state = MsgState.DownloadSuccess;
  184. msg.localFile = path;
  185. SqlUtil().updateLocalFile(msg.extraFile, path,
  186. isGroup: msg.channelType == ChatChannelType.Group.value);
  187. return path;
  188. } else {
  189. msg.state = MsgState.DownloadFailed;
  190. showToast(I18n.of(Constants.getCurrentContext()).server_error_tips);
  191. return null;
  192. }
  193. } catch (e) {
  194. msg.state = MsgState.DownloadFailed;
  195. showToast(I18n.of(Constants.getCurrentContext()).server_error_tips);
  196. print(response.toString());
  197. }
  198. }
  199. //机器人翻译,type 1文字,2语音
  200. Future<String> commitTranslateSource(
  201. int type, int sourceLang, int toLang, dynamic content) async {
  202. Response response;
  203. var myId = UserData().basicInfo.userId;
  204. Map<String, dynamic> data = {
  205. 'userid': myId,
  206. 'cType': type,
  207. 'ulanguage': sourceLang,
  208. 'tlanguage': toLang
  209. };
  210. data['sign'] = getSign(data);
  211. data['content'] = content;
  212. if (type == 2) {
  213. data['content'] = await MultipartFile.fromFile(content);
  214. }
  215. FormData formData = new FormData.fromMap(data);
  216. try {
  217. print('开始请求翻译$data ${uploadUrl + 'translate/transqt'}');
  218. response = await dio.post(rootUrl + '/translate/transqt', data: formData);
  219. Map resData = response.data;
  220. print('翻译结果$resData');
  221. if (resData['code'] == 0) {
  222. return resData['data'];
  223. }
  224. } catch (e) {
  225. print(e);
  226. }
  227. return null;
  228. }
  229. //获取群列表
  230. Future<List<GroupInfoModel>> getGroupList() async {
  231. Response response;
  232. var myId = UserData().basicInfo.userId;
  233. Map<String, dynamic> data = {
  234. 'userid': myId,
  235. };
  236. data['sign'] = getSign(data);
  237. try {
  238. print('开始请求群列表$data}');
  239. response = await dio.post(rootUrl + '/chat/queryusergroups', data: data);
  240. Map resData = response.data;
  241. if (resData['code'] == 0) {
  242. List resList = resData['data'];
  243. List<GroupInfoModel> groupList = [];
  244. for (var i = 0; i < resList.length; i++) {
  245. var info = GroupInfoServerModel.fromJson(resList[i]);
  246. groupList.add(GroupInfoModel.fromServerGroupInfo(info));
  247. }
  248. return groupList;
  249. }
  250. } catch (e) {
  251. print(e);
  252. }
  253. return null;
  254. }
  255. //获取多个群信息
  256. Future<List<GroupInfoModel>> getMoreGroupInfo(List<int> groupIds) async {
  257. Response response;
  258. var myId = UserData().basicInfo.userId;
  259. Map<String, dynamic> data = {
  260. 'userid': myId,
  261. };
  262. data['sign'] = getSign(data);
  263. data['groupids'] = groupIds;
  264. try {
  265. print('开始请求 getMoreGroupInfo $data');
  266. response = await dio.post(rootUrl + '/chat/querygroups', data: data);
  267. Map resData = response.data;
  268. if (resData['code'] == 0) {
  269. List data = resData['data'];
  270. print('群信息$data');
  271. if (data != null && data.length > 0) {
  272. List<GroupInfoModel> res = [];
  273. for (var each in data) {
  274. var info = GroupInfoServerModel.fromJson(each);
  275. res.add(GroupInfoModel.fromServerGroupInfo(info));
  276. }
  277. return res;
  278. }
  279. }
  280. } catch (e) {
  281. print(e);
  282. }
  283. return null;
  284. }
  285. //获取群信息
  286. Future<GroupInfoModel> getGroupInfo(int groupId) async {
  287. Response response;
  288. var myId = UserData().basicInfo.userId;
  289. Map<String, dynamic> data = {
  290. 'userid': myId,
  291. };
  292. data['sign'] = getSign(data);
  293. data['groupids'] = [groupId];
  294. try {
  295. print('开始请求群信息$data');
  296. response = await dio.post(rootUrl + '/chat/querygroups', data: data);
  297. Map resData = response.data;
  298. if (resData['code'] == 0) {
  299. List data = resData['data'];
  300. print('群信息$data');
  301. if (data != null && data.length > 0) {
  302. var info = GroupInfoServerModel.fromJson(data.first);
  303. return GroupInfoModel.fromServerGroupInfo(info);
  304. }
  305. }
  306. } catch (e) {
  307. print(e);
  308. }
  309. return null;
  310. }
  311. //获取群校验信息
  312. Future<Map> getGroupCheckInfo() async {
  313. Response response;
  314. var myId = UserData().basicInfo.userId;
  315. Map<String, dynamic> data = {
  316. 'userid': myId,
  317. };
  318. data['sign'] = getSign(data);
  319. try {
  320. print('请求获取群校验信息$data');
  321. response = await dio.post(rootUrl + '/chat/checkgroup', data: data);
  322. Map resData = response.data;
  323. if (resData['code'] == 0) {
  324. print('群校验信息${resData['data']}');
  325. return resData['data'];
  326. }
  327. } catch (e) {
  328. print(e);
  329. }
  330. return null;
  331. }
  332. }