vid_us_data_http_reader.dart 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367
  1. import 'dart:math';
  2. import 'dart:typed_data';
  3. import 'package:dio/dio.dart';
  4. //Raised when data is not ready.
  5. class NotReadyException implements Exception {}
  6. //Raised when errro happened during the downloading.
  7. class DownloadErrorException implements Exception {
  8. String _message;
  9. String get message => _message;
  10. DownloadErrorException(this._message);
  11. }
  12. typedef DownloadCallback = void Function(double progress, DownloadErrorException? error);
  13. class VidUsDataHttpReader {
  14. late Uint8List _underlyingData = Uint8List(0);
  15. late int _bufferSize = 0;
  16. late int _index;
  17. late bool _error;
  18. late String _errorMsg;
  19. late String _url;
  20. late int _minChunkSize;
  21. late Dio _dio;
  22. late double _downloadProgress = 0;
  23. DownloadCallback? downloadCallback;
  24. VidUsDataHttpReader(String url, {this.downloadCallback, int minChunkSize = 65536, int connectTimeout = 30000, int receiveTimeout = 30000}) {
  25. _minChunkSize = minChunkSize;
  26. _url = url;
  27. _error = false;
  28. _errorMsg = '';
  29. _index = 0;
  30. _dio = Dio();
  31. _dio.options.connectTimeout = connectTimeout;
  32. _dio.options.receiveTimeout = receiveTimeout;
  33. _startDownload();
  34. }
  35. ///Return the internal underlying data.
  36. Uint8List toBytes() {
  37. return _underlyingData;
  38. }
  39. ///Start the download in async way.
  40. void _startDownload() async {
  41. try {
  42. //get file info
  43. var fileSize = await _getFileSize();
  44. if (!_error) {
  45. _underlyingData = Uint8List(fileSize);
  46. if (fileSize < _minChunkSize) {
  47. await _downloadChunk(0, fileSize - 1);
  48. if (!_error && downloadCallback != null) {
  49. _downloadProgress = _bufferSize / fileSize;
  50. downloadCallback!(_downloadProgress, null);
  51. }else{
  52. throw DownloadErrorException(_errorMsg);
  53. }
  54. } else {
  55. var chunkSize = max(_minChunkSize, fileSize~/100) ;
  56. var last = fileSize % chunkSize;
  57. var chunkCount = fileSize ~/ chunkSize;
  58. for (var i = 0; i < chunkCount; i++) {
  59. var start = i * chunkSize;
  60. var end = start + chunkSize - 1;
  61. await _downloadChunk(start, end);
  62. if (_error) {
  63. throw DownloadErrorException(_errorMsg);
  64. }
  65. if (downloadCallback != null) {
  66. _downloadProgress = _bufferSize / fileSize;
  67. downloadCallback!(_downloadProgress, null);
  68. }
  69. }
  70. if (!_error && last > 0) {
  71. var start = chunkCount * chunkSize;
  72. var end = start + last - 1;
  73. await _downloadChunk(start, end);
  74. if (!_error && downloadCallback != null) {
  75. _downloadProgress = _bufferSize / fileSize;
  76. downloadCallback!(_downloadProgress, null);
  77. }else{
  78. throw DownloadErrorException(_errorMsg);
  79. }
  80. }
  81. }
  82. }
  83. } on DownloadErrorException catch (e) {
  84. if (!_error && downloadCallback != null) {
  85. downloadCallback!(_downloadProgress, e);
  86. }
  87. } on Exception catch (e) {
  88. _error = true;
  89. _errorMsg = e.toString();
  90. if (!_error && downloadCallback != null) {
  91. downloadCallback!(_downloadProgress, DownloadErrorException(_errorMsg));
  92. }
  93. }
  94. }
  95. Future<int> _getFileSize() async {
  96. var response = await _dio.head(_url);
  97. if (response.statusCode != 200) {
  98. _error = true;
  99. _errorMsg = 'Get file info error.';
  100. return 0;
  101. } else {
  102. var contentLength = response.headers.value(Headers.contentLengthHeader);
  103. if (contentLength == null) {
  104. _error = true;
  105. _errorMsg = 'No Content-Length from headers.';
  106. return 0;
  107. } else {
  108. var fileSize = int.parse(contentLength);
  109. return fileSize;
  110. }
  111. }
  112. }
  113. Future _downloadChunk(int chunkStart, int chunkEnd) async {
  114. var response = await _dio.get<List<int>>(_url, options: Options(responseType: ResponseType.bytes, headers: {"range": "bytes=$chunkStart-$chunkEnd"}));
  115. if (response.statusCode == 206) {
  116. var downloadedData = response.data;
  117. if (downloadedData == null) {
  118. _error = true;
  119. _errorMsg = 'Download chunk error.';
  120. } else {
  121. //Update the buffer.
  122. _underlyingData.setAll(_bufferSize, downloadedData);
  123. _bufferSize += downloadedData.length;
  124. }
  125. } else {
  126. _error = true;
  127. _errorMsg = 'Target url does not support download with chunks.';
  128. }
  129. }
  130. //Close the reader and its http connection.
  131. void close() {
  132. _dio.close(force: true);
  133. }
  134. ///Read int value from binary data(little endian),
  135. int readInt([int index = -1]) {
  136. if (_error) {
  137. throw DownloadErrorException(_errorMsg);
  138. }
  139. if (index == -1) {
  140. if ((_bufferSize - _index) < 4) {
  141. throw NotReadyException();
  142. }
  143. var intData = _underlyingData.buffer.asByteData(_index, 4);
  144. _index += 4;
  145. return intData.getInt32(0, Endian.little);
  146. } else {
  147. if ((_bufferSize - index) < 4) {
  148. throw NotReadyException();
  149. }
  150. var intData = _underlyingData.buffer.asByteData(index, 4);
  151. _index = index + 4;
  152. return intData.getInt32(0, Endian.little);
  153. }
  154. }
  155. ///Read string from the binary data, the string format is utf-16, which is unicode in C#
  156. String readString([int index = -1]) {
  157. if (_error) {
  158. throw DownloadErrorException(_errorMsg);
  159. }
  160. int dataLength;
  161. if (index == -1) {
  162. dataLength = readInt();
  163. if ((_bufferSize - _index) < dataLength) {
  164. throw NotReadyException();
  165. }
  166. } else {
  167. dataLength = readInt(index);
  168. if ((_bufferSize - index) < dataLength) {
  169. throw NotReadyException();
  170. }
  171. }
  172. var stringData = _underlyingData.buffer.asUint8List(_index, dataLength);
  173. _index += dataLength;
  174. var unicodeStringData = Uint8List.fromList(stringData.toList()).buffer.asUint16List();
  175. return String.fromCharCodes(unicodeStringData);
  176. }
  177. ///Read int16 value from binary data(little endian)
  178. int readInt16([int index = -1]) {
  179. if (_error) {
  180. throw DownloadErrorException(_errorMsg);
  181. }
  182. if (index == -1) {
  183. if ((_bufferSize - _index) < 2) {
  184. throw NotReadyException();
  185. }
  186. var intData = _underlyingData.buffer.asByteData(_index, 2);
  187. _index += 2;
  188. return intData.getInt16(0, Endian.little);
  189. } else {
  190. if ((_bufferSize - index) < 2) {
  191. throw NotReadyException();
  192. }
  193. var intData = _underlyingData.buffer.asByteData(index, 2);
  194. _index = index + 2;
  195. return intData.getInt16(0, Endian.little);
  196. }
  197. }
  198. ///Read int64 value from binary data(little endian)
  199. ///this method is not support in web platform use readInt64V2 instead.
  200. int readInt64([int index = -1]) {
  201. if (_error) {
  202. throw DownloadErrorException(_errorMsg);
  203. }
  204. if (index == -1) {
  205. if ((_bufferSize - _index) < 8) {
  206. throw NotReadyException();
  207. }
  208. var intData = _underlyingData.buffer.asByteData(_index, 8);
  209. _index += 8;
  210. return intData.getInt64(0, Endian.little);
  211. } else {
  212. if ((_bufferSize - index) < 8) {
  213. throw NotReadyException();
  214. }
  215. var intData = _underlyingData.buffer.asByteData(index, 8);
  216. _index = index + 8;
  217. return intData.getInt64(0, Endian.little);
  218. }
  219. }
  220. ///Read int64 value from binary data(little endian)
  221. ///this method use two int32 to support read int64 on web platform
  222. int readInt64V2([int index = -1]) {
  223. if (_error) {
  224. throw DownloadErrorException(_errorMsg);
  225. }
  226. if (index == -1) {
  227. if ((_bufferSize - _index) < 8) {
  228. throw NotReadyException();
  229. }
  230. int low = readInt();
  231. int high = readInt();
  232. int value = high >> 32;
  233. value |= low;
  234. return value;
  235. } else {
  236. if ((_bufferSize - index) < 8) {
  237. throw NotReadyException();
  238. }
  239. int low = readInt(index);
  240. //Read the next int.
  241. int high = readInt();
  242. int value = high >> 32;
  243. value |= low;
  244. return value;
  245. }
  246. }
  247. ///Read float value from binary data(little endian)
  248. double readFloat([int index = -1]) {
  249. if (_error) {
  250. throw DownloadErrorException(_errorMsg);
  251. }
  252. if (index == -1) {
  253. if ((_bufferSize - _index) < 4) {
  254. throw NotReadyException();
  255. }
  256. var floatData = _underlyingData.buffer.asByteData(_index, 4);
  257. _index += 4;
  258. return floatData.getFloat32(0, Endian.little);
  259. } else {
  260. if ((_bufferSize - index) < 4) {
  261. throw NotReadyException();
  262. }
  263. var floatData = _underlyingData.buffer.asByteData(index, 4);
  264. _index = index + 4;
  265. return floatData.getFloat32(0, Endian.little);
  266. }
  267. }
  268. ///Read double value from binary data(little endian)
  269. double readDouble([int index = -1]) {
  270. if (_error) {
  271. throw DownloadErrorException(_errorMsg);
  272. }
  273. if (index == -1) {
  274. if ((_bufferSize - _index) < 8) {
  275. throw NotReadyException();
  276. }
  277. var floatData = _underlyingData.buffer.asByteData(_index, 8);
  278. _index += 8;
  279. return floatData.getFloat64(0, Endian.little);
  280. } else {
  281. if ((_bufferSize - index) < 8) {
  282. throw NotReadyException();
  283. }
  284. var floatData = _underlyingData.buffer.asByteData(index, 8);
  285. _index = index + 8;
  286. return floatData.getFloat64(0, Endian.little);
  287. }
  288. }
  289. ///Read bool value from binary data
  290. bool readBool([int index = -1]) {
  291. if (_error) {
  292. throw DownloadErrorException(_errorMsg);
  293. }
  294. if (index == -1) {
  295. if ((_bufferSize - _index) < 1) {
  296. throw NotReadyException();
  297. }
  298. var boolData = _underlyingData.buffer.asByteData(_index, 1);
  299. _index++;
  300. return boolData.getInt8(0) == 1;
  301. } else {
  302. if ((_bufferSize - index) < 1) {
  303. throw NotReadyException();
  304. }
  305. var boolData = _underlyingData.buffer.asByteData(index, 1);
  306. _index = index + 1;
  307. return boolData.getInt8(0) == 1;
  308. }
  309. }
  310. ///Read byte value from binary data
  311. int readByte([int index = -1]) {
  312. if (_error) {
  313. throw DownloadErrorException(_errorMsg);
  314. }
  315. if (index == -1) {
  316. if ((_bufferSize - _index) < 1) {
  317. throw NotReadyException();
  318. }
  319. var byteData = _underlyingData.buffer.asByteData(_index, 1);
  320. _index++;
  321. return byteData.getInt8(0);
  322. } else {
  323. if ((_bufferSize - index) < 1) {
  324. throw NotReadyException();
  325. }
  326. var byteData = _underlyingData.buffer.asByteData(index, 1);
  327. _index = index + 1;
  328. return byteData.getInt8(0);
  329. }
  330. }
  331. ///Read bytes from the binary data.
  332. Uint8List readBytes([int index = -1]) {
  333. if (_error) {
  334. throw DownloadErrorException(_errorMsg);
  335. }
  336. var dataLength = readInt(index);
  337. if ((_bufferSize - _index) < dataLength) {
  338. throw NotReadyException();
  339. }
  340. var bytes = _underlyingData.buffer.asUint8List(_index, dataLength);
  341. _index += dataLength;
  342. return bytes;
  343. }
  344. }