|
|
using System;
|
|
|
using System.Collections.Concurrent;
|
|
|
using System.Collections.Generic;
|
|
|
using System.IO;
|
|
|
using System.Net.WebSockets;
|
|
|
using System.Threading;
|
|
|
using System.Threading.Tasks;
|
|
|
using Microsoft.Extensions.Logging;
|
|
|
using Newtonsoft.Json;
|
|
|
using Yavsc.Models;
|
|
|
using Yavsc.Models.FileSystem;
|
|
|
|
|
|
namespace Yavsc.ViewModels.Streaming
|
|
|
{
|
|
|
public class LiveCastClient {
|
|
|
public string UserName { get; set; }
|
|
|
public WebSocket Socket { get; set; }
|
|
|
}
|
|
|
|
|
|
public class LiveEntryViewModel {
|
|
|
public string UserName { get; set; }
|
|
|
public string FlowId { get; set; }
|
|
|
}
|
|
|
|
|
|
public class LiveCastHandler : IDisposable
|
|
|
{
|
|
|
|
|
|
public WebSocket Socket { get; set; }
|
|
|
public ConcurrentDictionary<string, WebSocket> Listeners { get; set; } = new ConcurrentDictionary<string, WebSocket>();
|
|
|
|
|
|
public CancellationTokenSource TokenSource { get; set; } = new CancellationTokenSource();
|
|
|
|
|
|
public void Dispose()
|
|
|
{
|
|
|
}
|
|
|
|
|
|
public async Task<FileRecievedInfo> ReceiveUserFile(ApplicationUser user, ILogger logger, string root, Queue<ArraySegment<byte>> queue, string destFileName, string contentType, Func<bool> isEndOfInput)
|
|
|
{
|
|
|
// TODO lock user's disk usage for this scope,
|
|
|
// this process is not safe at concurrent access.
|
|
|
long usage = user.DiskUsage;
|
|
|
|
|
|
var item = new FileRecievedInfo();
|
|
|
item.FileName = Yavsc.Abstract.FileSystem.AbstractFileSystemHelpers.FilterFileName (destFileName);
|
|
|
item.MimeType = contentType;
|
|
|
item.DestDir = root;
|
|
|
var fi = new FileInfo(Path.Combine(root, item.FileName));
|
|
|
if (fi.Exists)
|
|
|
{
|
|
|
item.Overriden = true;
|
|
|
usage -= fi.Length;
|
|
|
}
|
|
|
logger.LogInformation("Opening the file");
|
|
|
using (var dest = fi.Open(FileMode.Create, FileAccess.Write, FileShare.Read))
|
|
|
{
|
|
|
logger.LogInformation("Appening to file");
|
|
|
while (!isEndOfInput() || queue.Count>0)
|
|
|
{
|
|
|
if (queue.Count>0) {
|
|
|
var buffer = queue.Dequeue();
|
|
|
int count = buffer.Array[0]*256*1024 +buffer.Array[1]*1024+buffer.Array[2]*256 + buffer.Array[3];
|
|
|
|
|
|
if (count >0 && count <= Constants.WebSocketsMaxBufLen
|
|
|
&& buffer.Array.Length >= count+4) {
|
|
|
logger.LogInformation($"writing {count} bytes from {buffer.Array.Length}.");
|
|
|
|
|
|
await dest.WriteAsync(buffer.Array, 4, count);
|
|
|
logger.LogInformation($"wrote {count} bytes.");
|
|
|
usage += count;
|
|
|
}
|
|
|
else {
|
|
|
var packetInfo = JsonConvert.SerializeObject(buffer);
|
|
|
logger.LogError($"didn´t wrote {count} bytes from {buffer.Array.Length}!\n{packetInfo}");
|
|
|
}
|
|
|
}
|
|
|
if (usage >= user.DiskQuota) break;
|
|
|
if (queue.Count==0 && !isEndOfInput()) {
|
|
|
logger.LogInformation($"Waitting 200ms.");
|
|
|
await Task.Delay(100);
|
|
|
logger.LogInformation($"Done waiting");
|
|
|
}
|
|
|
}
|
|
|
user.DiskUsage = usage;
|
|
|
dest.Close();
|
|
|
}
|
|
|
if (usage >= user.DiskQuota) {
|
|
|
item.QuotaOffensed = true;
|
|
|
}
|
|
|
user.DiskUsage = usage;
|
|
|
return item;
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
} |