Surface/lib/providers/sn_attachment.dart

235 lines
6.6 KiB
Dart
Raw Normal View History

2024-11-09 17:04:39 +00:00
import 'dart:collection';
2024-11-11 13:48:50 +00:00
import 'dart:math' as math;
2024-11-09 17:04:39 +00:00
import 'dart:typed_data';
import 'package:dio/dio.dart';
2024-11-09 04:04:03 +00:00
import 'package:flutter/widgets.dart';
2024-11-09 17:04:39 +00:00
import 'package:cross_file/cross_file.dart';
2024-11-09 04:04:03 +00:00
import 'package:provider/provider.dart';
import 'package:surface/providers/sn_network.dart';
import 'package:surface/types/attachment.dart';
2024-11-09 17:04:39 +00:00
const kConcurrentUploadChunks = 5;
2024-11-09 04:04:03 +00:00
class SnAttachmentProvider {
2024-11-09 11:32:21 +00:00
late final SnNetworkProvider _sn;
2024-11-09 04:04:03 +00:00
final Map<String, SnAttachment> _cache = {};
SnAttachmentProvider(BuildContext context) {
2024-11-09 11:32:21 +00:00
_sn = context.read<SnNetworkProvider>();
2024-11-09 04:04:03 +00:00
}
2024-11-17 16:55:39 +00:00
void putCache(Iterable<SnAttachment> items, {bool noCheck = false}) {
for (final item in items) {
if ((item.isAnalyzed && item.isUploaded) || noCheck) {
_cache[item.rid] = item;
}
}
}
2024-11-09 04:04:03 +00:00
Future<SnAttachment> getOne(String rid, {noCache = false}) async {
if (!noCache && _cache.containsKey(rid)) {
return _cache[rid]!;
}
2024-11-09 11:32:21 +00:00
final resp = await _sn.client.get('/cgi/uc/attachments/$rid/meta');
2024-11-09 04:04:03 +00:00
final out = SnAttachment.fromJson(resp.data);
2024-11-17 16:55:39 +00:00
if (out.isAnalyzed && out.isUploaded) {
_cache[rid] = out;
}
2024-11-09 04:04:03 +00:00
return out;
}
2024-12-21 16:48:06 +00:00
Future<List<SnAttachment?>> getMultiple(List<String> rids, {noCache = false}) async {
2024-11-17 16:55:39 +00:00
final result = List<SnAttachment?>.filled(rids.length, null);
final Map<String, int> randomMapping = {};
for (int i = 0; i < rids.length; i++) {
final rid = rids[i];
if (noCache || !_cache.containsKey(rid)) {
randomMapping[rid] = i;
} else {
result[i] = _cache[rid]!;
}
2024-11-09 04:04:03 +00:00
}
2024-11-17 16:55:39 +00:00
final pendingFetch = randomMapping.keys;
2024-11-09 04:04:03 +00:00
2024-11-17 16:55:39 +00:00
if (pendingFetch.isNotEmpty) {
final resp = await _sn.client.get(
'/cgi/uc/attachments',
queryParameters: {
'take': pendingFetch.length,
'id': pendingFetch.join(','),
},
);
2024-12-21 16:48:06 +00:00
final out = resp.data['data'].map((e) => e['id'] == 0 ? null : SnAttachment.fromJson(e)).toList();
2024-11-09 04:04:03 +00:00
2024-11-17 16:55:39 +00:00
for (final item in out) {
if (item == null) continue;
2024-11-17 16:55:39 +00:00
if (item.isAnalyzed && item.isUploaded) {
_cache[item.rid] = item;
}
result[randomMapping[item.rid]!] = item;
}
2024-11-09 04:04:03 +00:00
}
2024-11-12 16:25:02 +00:00
2024-11-17 16:55:39 +00:00
return result;
2024-11-09 04:04:03 +00:00
}
2024-11-09 17:04:39 +00:00
2024-12-21 16:48:06 +00:00
static Map<String, String> mimetypeOverrides = {'mov': 'video/quicktime', 'mp4': 'video/mp4'};
2024-11-09 17:04:39 +00:00
Future<SnAttachment> directUploadOne(
Uint8List data,
String filename,
String pool,
Map<String, dynamic>? metadata, {
String? mimetype,
Function(double progress)? onProgress,
}) async {
final filePayload = MultipartFile.fromBytes(data, filename: filename);
2024-12-21 16:48:06 +00:00
final fileAlt = filename.contains('.') ? filename.substring(0, filename.lastIndexOf('.')) : filename;
final fileExt = filename.substring(filename.lastIndexOf('.') + 1).toLowerCase();
2024-11-09 17:04:39 +00:00
String? mimetypeOverride;
if (mimetype != null) {
mimetypeOverride = mimetype;
} else if (mimetypeOverrides.keys.contains(fileExt)) {
mimetypeOverride = mimetypeOverrides[fileExt];
}
final formData = FormData.fromMap({
'alt': fileAlt,
'file': filePayload,
'pool': pool,
'metadata': metadata,
if (mimetypeOverride != null) 'mimetype': mimetypeOverride,
});
final resp = await _sn.client.post(
'/cgi/uc/attachments',
data: formData,
onSendProgress: (count, total) {
if (onProgress != null) {
onProgress(count / total);
}
},
);
return SnAttachment.fromJson(resp.data);
}
Future<(SnAttachment, int)> chunkedUploadInitialize(
int size,
String filename,
String pool,
2024-11-21 16:28:29 +00:00
Map<String, dynamic>? metadata, {
String? mimetype,
}) async {
2024-12-21 16:48:06 +00:00
final fileAlt = filename.contains('.') ? filename.substring(0, filename.lastIndexOf('.')) : filename;
final fileExt = filename.substring(filename.lastIndexOf('.') + 1).toLowerCase();
2024-11-09 17:04:39 +00:00
String? mimetypeOverride;
2024-11-21 16:28:29 +00:00
if (mimetype == null && mimetypeOverrides.keys.contains(fileExt)) {
2024-11-09 17:04:39 +00:00
mimetypeOverride = mimetypeOverrides[fileExt];
2024-11-21 16:28:29 +00:00
} else {
mimetypeOverride = mimetype;
2024-11-09 17:04:39 +00:00
}
final resp = await _sn.client.post('/cgi/uc/attachments/multipart', data: {
'alt': fileAlt,
'name': filename,
'pool': pool,
'metadata': metadata,
'size': size,
if (mimetypeOverride != null) 'mimetype': mimetypeOverride,
});
2024-12-21 16:48:06 +00:00
return (SnAttachment.fromJson(resp.data['meta']), resp.data['chunk_size'] as int);
2024-11-09 17:04:39 +00:00
}
Future<SnAttachment> _chunkedUploadOnePart(
Uint8List data,
String rid,
String cid, {
Function(double progress)? onProgress,
}) async {
final resp = await _sn.client.post(
'/cgi/uc/attachments/multipart/$rid/$cid',
data: data,
options: Options(headers: {'Content-Type': 'application/octet-stream'}),
onSendProgress: (count, total) {
if (onProgress != null) {
onProgress(count / total);
}
},
);
return SnAttachment.fromJson(resp.data);
}
Future<SnAttachment> chunkedUploadParts(
XFile file,
SnAttachment place,
int chunkSize, {
Function(double progress)? onProgress,
}) async {
final Map<String, dynamic> chunks = place.fileChunks ?? {};
var currentTask = 0;
final queue = Queue<Future<void>>();
final activeTasks = <Future<void>>[];
for (final entry in chunks.entries) {
queue.add(() async {
final beginCursor = entry.value * chunkSize;
2024-11-11 13:48:50 +00:00
final endCursor = math.min<int>(
(entry.value + 1) * chunkSize,
await file.length(),
);
2024-12-21 16:48:06 +00:00
final data = Uint8List.fromList(await file.openRead(beginCursor, endCursor).expand((chunk) => chunk).toList());
2024-11-09 17:04:39 +00:00
place = await _chunkedUploadOnePart(
data,
place.rid,
entry.key,
);
2024-12-21 16:48:06 +00:00
final overallProgress = currentTask / chunks.length;
onProgress?.call(overallProgress);
2024-11-09 17:04:39 +00:00
currentTask++;
}());
}
while (queue.isNotEmpty || activeTasks.isNotEmpty) {
while (activeTasks.length < kConcurrentUploadChunks && queue.isNotEmpty) {
final task = queue.removeFirst();
activeTasks.add(task);
task.then((_) => activeTasks.remove(task));
}
if (activeTasks.isNotEmpty) {
await Future.any(activeTasks);
}
}
return place;
}
Future<SnAttachment> updateOne(
2024-12-26 15:01:00 +00:00
int id, {
String? alt,
String? thumbnail,
Map<String, dynamic>? metadata,
bool? isIndexable,
}) async {
final resp = await _sn.client.put('/cgi/uc/attachments/$id', data: {
'alt': alt,
2024-12-26 15:01:00 +00:00
'thumbnail': thumbnail,
'metadata': metadata,
2024-12-26 15:01:00 +00:00
'is_indexable': isIndexable,
});
return SnAttachment.fromJson(resp.data);
}
2024-11-09 04:04:03 +00:00
}