mirror of
https://github.com/Polaris-Entertainment/bytefy.git
synced 2026-04-11 10:29:37 +00:00
Test queue
This commit is contained in:
@@ -0,0 +1,48 @@
|
|||||||
|
using System.Collections.Concurrent;
|
||||||
|
using ImageMagick;
|
||||||
|
|
||||||
|
namespace bytefy.image;
|
||||||
|
|
||||||
|
public class ConversionTask
|
||||||
|
{
|
||||||
|
public byte[] ImageData { get; set; }
|
||||||
|
public MagickFormat Format { get; set; }
|
||||||
|
}
|
||||||
|
|
||||||
|
public class ConversionQueueService : BackgroundService
|
||||||
|
{
|
||||||
|
private readonly ConcurrentQueue<(ConversionTask Task, TaskCompletionSource<(byte[], string)> CompletionSource)> _queue = new();
|
||||||
|
|
||||||
|
public void QueueConversion(ConversionTask task, TaskCompletionSource<(byte[], string)> completionSource)
|
||||||
|
{
|
||||||
|
_queue.Enqueue((task, completionSource));
|
||||||
|
}
|
||||||
|
|
||||||
|
protected override async Task ExecuteAsync(CancellationToken stoppingToken)
|
||||||
|
{
|
||||||
|
while (!stoppingToken.IsCancellationRequested)
|
||||||
|
{
|
||||||
|
if (_queue.TryDequeue(out var item))
|
||||||
|
{
|
||||||
|
var result = await ProcessConversionAsync(item.Task);
|
||||||
|
item.CompletionSource.SetResult(result);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
await Task.Delay(50, stoppingToken); // Shorten the delay to check the queue more frequently
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private Task<(byte[], string)> ProcessConversionAsync(ConversionTask task)
|
||||||
|
{
|
||||||
|
using var magickImage = new MagickImage(task.ImageData);
|
||||||
|
magickImage.Format = task.Format;
|
||||||
|
var resultStream = new MemoryStream();
|
||||||
|
magickImage.Write(resultStream);
|
||||||
|
resultStream.Position = 0;
|
||||||
|
|
||||||
|
var mimeType = MimeTypes.MimeTypeMap.GetMimeType($"image/{task.Format.ToString().ToLower()}");
|
||||||
|
return Task.FromResult((resultStream.ToArray(), mimeType));
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -1,42 +1,44 @@
|
|||||||
|
using bytefy.image;
|
||||||
using ImageMagick;
|
using ImageMagick;
|
||||||
using Microsoft.AspNetCore.Antiforgery;
|
using Microsoft.AspNetCore.Antiforgery;
|
||||||
|
|
||||||
var builder = WebApplication.CreateBuilder(args);
|
var builder = WebApplication.CreateBuilder(args);
|
||||||
builder.Services.AddAntiforgery(options => options.HeaderName = "2311d8d8-607d-4747-8939-1bde65643254");
|
builder.Services.AddAntiforgery(options => options.HeaderName = "2311d8d8-607d-4747-8939-1bde65643254");
|
||||||
|
builder.Services.AddSingleton<ConversionQueueService>();
|
||||||
|
builder.Services.AddHostedService(provider => provider.GetRequiredService<ConversionQueueService>());
|
||||||
|
|
||||||
var app = builder.Build();
|
var app = builder.Build();
|
||||||
|
|
||||||
app.UseAntiforgery();
|
app.UseAntiforgery();
|
||||||
|
|
||||||
|
var conversionQueue = app.Services.GetRequiredService<ConversionQueueService>();
|
||||||
|
|
||||||
app.MapPost("/convert/{format}", async (IFormFile image, string format) =>
|
app.MapPost("/convert/{format}", async (IFormFile image, string format) =>
|
||||||
{
|
{
|
||||||
try
|
if (!Enum.TryParse(format, true, out MagickFormat magickFormat) || magickFormat == MagickFormat.Unknown)
|
||||||
|
return Results.BadRequest("Invalid format");
|
||||||
|
|
||||||
|
if (image == null || image.Length == 0)
|
||||||
|
return Results.BadRequest("No image provided");
|
||||||
|
|
||||||
|
if (image.Length > 20 * 1024 * 1024)
|
||||||
|
throw new Exception("Image size too large");
|
||||||
|
|
||||||
|
using var memoryStream = new MemoryStream();
|
||||||
|
await image.CopyToAsync(memoryStream);
|
||||||
|
|
||||||
|
var conversionTask = new ConversionTask
|
||||||
{
|
{
|
||||||
if (!Enum.TryParse(format, true, out MagickFormat magickFormat) || magickFormat == MagickFormat.Unknown)
|
ImageData = memoryStream.ToArray(),
|
||||||
return Results.BadRequest("Invalid format");
|
Format = magickFormat
|
||||||
|
};
|
||||||
|
|
||||||
if (image == null || image.Length == 0)
|
var tcs = new TaskCompletionSource<(byte[], string)>();
|
||||||
return Results.BadRequest("No image provided");
|
conversionQueue.QueueConversion(conversionTask, tcs);
|
||||||
|
|
||||||
if (image.Length > 20 * 1024 * 1024)
|
var (imageData, mimeType) = await tcs.Task;
|
||||||
throw new Exception("Image size too large");
|
|
||||||
|
|
||||||
using var memoryStream = new MemoryStream();
|
return Results.File(new MemoryStream(imageData), mimeType, $"{Path.GetFileNameWithoutExtension(image.FileName)}.{magickFormat.ToString().ToLower()}");
|
||||||
await image.CopyToAsync(memoryStream);
|
|
||||||
var magickImage = new MagickImage(memoryStream.ToArray());
|
|
||||||
magickImage.Format = magickFormat;
|
|
||||||
|
|
||||||
var resultStream = new MemoryStream();
|
|
||||||
magickImage.Write(resultStream);
|
|
||||||
resultStream.Position = 0;
|
|
||||||
|
|
||||||
var mimeType = MimeTypes.MimeTypeMap.GetMimeType($"image/{magickFormat.ToString().ToLower()}");
|
|
||||||
var fileName = Path.GetFileNameWithoutExtension(image.FileName);
|
|
||||||
|
|
||||||
return Results.File(resultStream, mimeType, fileDownloadName: $"{fileName}.{magickFormat.ToString().ToLower()}");
|
|
||||||
}
|
|
||||||
catch (NotSupportedException) {
|
|
||||||
return Results.BadRequest();
|
|
||||||
}
|
|
||||||
});
|
});
|
||||||
|
|
||||||
app.MapGet("/antiforgery/token", (IAntiforgery forgeryService, HttpContext context) =>
|
app.MapGet("/antiforgery/token", (IAntiforgery forgeryService, HttpContext context) =>
|
||||||
@@ -46,4 +48,12 @@ app.MapGet("/antiforgery/token", (IAntiforgery forgeryService, HttpContext conte
|
|||||||
return TypedResults.Content(xsrfToken, "text/plain");
|
return TypedResults.Content(xsrfToken, "text/plain");
|
||||||
});
|
});
|
||||||
|
|
||||||
app.Run();
|
app.MapGet("/formats", () =>
|
||||||
|
{
|
||||||
|
var formats = Enum.GetNames<MagickFormat>().ToList();
|
||||||
|
|
||||||
|
formats.Remove("Unknown");
|
||||||
|
return Results.Ok(formats);
|
||||||
|
});
|
||||||
|
|
||||||
|
app.Run();
|
||||||
|
|||||||
@@ -5,7 +5,7 @@
|
|||||||
<Nullable>enable</Nullable>
|
<Nullable>enable</Nullable>
|
||||||
<ImplicitUsings>enable</ImplicitUsings>
|
<ImplicitUsings>enable</ImplicitUsings>
|
||||||
<InvariantGlobalization>true</InvariantGlobalization>
|
<InvariantGlobalization>true</InvariantGlobalization>
|
||||||
<PublishAot>true</PublishAot>
|
<PublishAot>false</PublishAot>
|
||||||
<DockerDefaultTargetOS>Linux</DockerDefaultTargetOS>
|
<DockerDefaultTargetOS>Linux</DockerDefaultTargetOS>
|
||||||
</PropertyGroup>
|
</PropertyGroup>
|
||||||
|
|
||||||
|
|||||||
@@ -1,10 +1,10 @@
|
|||||||
import { Routes } from '@angular/router';
|
import { Routes } from '@angular/router';
|
||||||
import { AsciiToTextComponent } from '../tools/ascii-to-text/ascii-to-text.component';
|
import { AsciiToTextComponent } from '../tools/client-side/ascii-to-text/ascii-to-text.component';
|
||||||
import { GuidComponent } from '../tools/guid/guid.component';
|
import { GuidComponent } from '../tools/client-side/guid/guid.component';
|
||||||
import { Base64ConverterComponent } from '../tools/base64-converter/base64-converter.component';
|
import { Base64ConverterComponent } from '../tools/client-side/base64-converter/base64-converter.component';
|
||||||
import { JwtToJsonComponent } from '../tools/jwt-to-json/jwt-to-json.component';
|
import { JwtToJsonComponent } from '../tools/client-side/jwt-to-json/jwt-to-json.component';
|
||||||
import { TextToCronComponent } from '../tools/text-to-cron/text-to-cron.component';
|
import { TextToCronComponent } from '../tools/client-side/text-to-cron/text-to-cron.component';
|
||||||
import { DdsToPngComponent } from '../tools/dds-to-png/dds-to-png.component';
|
import { DdsToPngComponent } from '../tools/client-side/dds-to-png/dds-to-png.component';
|
||||||
|
|
||||||
export const routes: Routes = [
|
export const routes: Routes = [
|
||||||
{
|
{
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
import { Component } from '@angular/core';
|
import { Component } from '@angular/core';
|
||||||
import { DualTextareaComponent } from '../../app/shared/dual-textarea/dual-textarea.component';
|
import { DualTextareaComponent } from '../../../app/shared/dual-textarea/dual-textarea.component';
|
||||||
|
|
||||||
@Component({
|
@Component({
|
||||||
selector: 'app-ascii-to-text',
|
selector: 'app-ascii-to-text',
|
||||||
@@ -1,5 +1,5 @@
|
|||||||
import { Component } from '@angular/core';
|
import { Component } from '@angular/core';
|
||||||
import { DualTextareaComponent } from '../../app/shared/dual-textarea/dual-textarea.component';
|
import { DualTextareaComponent } from '../../../app/shared/dual-textarea/dual-textarea.component';
|
||||||
|
|
||||||
@Component({
|
@Component({
|
||||||
selector: 'app-base64-converter',
|
selector: 'app-base64-converter',
|
||||||
@@ -1,5 +1,5 @@
|
|||||||
import { Component, OnInit } from '@angular/core';
|
import { Component, OnInit } from '@angular/core';
|
||||||
import { FileConverterComponent } from '../../app/shared/upload/file-converter.component';
|
import { FileConverterComponent } from '../../../app/shared/upload/file-converter.component';
|
||||||
import { DdsToPngService } from './dds-to-png.service';
|
import { DdsToPngService } from './dds-to-png.service';
|
||||||
|
|
||||||
interface ProcessedFile {
|
interface ProcessedFile {
|
||||||
@@ -1,5 +1,5 @@
|
|||||||
import { Component } from '@angular/core';
|
import { Component } from '@angular/core';
|
||||||
import { DualTextareaComponent } from '../../app/shared/dual-textarea/dual-textarea.component';
|
import { DualTextareaComponent } from '../../../app/shared/dual-textarea/dual-textarea.component';
|
||||||
|
|
||||||
@Component({
|
@Component({
|
||||||
selector: 'app-jwt-to-json',
|
selector: 'app-jwt-to-json',
|
||||||
@@ -1,5 +1,5 @@
|
|||||||
import { Component } from '@angular/core';
|
import { Component } from '@angular/core';
|
||||||
import { DualTextareaComponent } from '../../app/shared/dual-textarea/dual-textarea.component';
|
import { DualTextareaComponent } from '../../../app/shared/dual-textarea/dual-textarea.component';
|
||||||
|
|
||||||
@Component({
|
@Component({
|
||||||
selector: 'app-text-to-cron',
|
selector: 'app-text-to-cron',
|
||||||
@@ -0,0 +1,31 @@
|
|||||||
|
import { Injectable } from '@angular/core';
|
||||||
|
import { HttpClient, HttpHeaders } from '@angular/common/http';
|
||||||
|
import { Observable, tap } from 'rxjs';
|
||||||
|
|
||||||
|
@Injectable({
|
||||||
|
providedIn: 'root'
|
||||||
|
})
|
||||||
|
export class ImageService {
|
||||||
|
private baseUrl = 'http://localhost:1337'; // replace with your API base URL
|
||||||
|
|
||||||
|
constructor(private http: HttpClient) { }
|
||||||
|
|
||||||
|
convertImage(image: File, format: string): Observable<any> {
|
||||||
|
const formData = new FormData();
|
||||||
|
formData.append('image', image);
|
||||||
|
|
||||||
|
let imgToken = localStorage.getItem('imgToken');
|
||||||
|
|
||||||
|
const headers = new HttpHeaders({
|
||||||
|
'2311d8d8-607d-4747-8939-1bde65643254': imgToken!
|
||||||
|
});
|
||||||
|
|
||||||
|
return this.http.post(`${this.baseUrl}/convert/${format}`, formData, { headers });
|
||||||
|
}
|
||||||
|
|
||||||
|
seteAntiforgeryToken(): void {
|
||||||
|
this.http.get<string>(`${this.baseUrl}/antiforgery/token`, { responseType: 'text' as 'json' }).pipe(
|
||||||
|
tap(token => localStorage.setItem('imgToken', token))
|
||||||
|
)
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,3 @@
|
|||||||
|
<p>
|
||||||
|
image-converter works!
|
||||||
|
</p>
|
||||||
@@ -0,0 +1,28 @@
|
|||||||
|
/* tslint:disable:no-unused-variable */
|
||||||
|
import { async, ComponentFixture, TestBed } from '@angular/core/testing';
|
||||||
|
import { By } from '@angular/platform-browser';
|
||||||
|
import { DebugElement } from '@angular/core';
|
||||||
|
|
||||||
|
import { ImageConverterComponent } from './image-converter.component';
|
||||||
|
|
||||||
|
describe('ImageConverterComponent', () => {
|
||||||
|
let component: ImageConverterComponent;
|
||||||
|
let fixture: ComponentFixture<ImageConverterComponent>;
|
||||||
|
|
||||||
|
beforeEach(async(() => {
|
||||||
|
TestBed.configureTestingModule({
|
||||||
|
declarations: [ ImageConverterComponent ]
|
||||||
|
})
|
||||||
|
.compileComponents();
|
||||||
|
}));
|
||||||
|
|
||||||
|
beforeEach(() => {
|
||||||
|
fixture = TestBed.createComponent(ImageConverterComponent);
|
||||||
|
component = fixture.componentInstance;
|
||||||
|
fixture.detectChanges();
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should create', () => {
|
||||||
|
expect(component).toBeTruthy();
|
||||||
|
});
|
||||||
|
});
|
||||||
@@ -0,0 +1,10 @@
|
|||||||
|
import { Component, OnInit } from '@angular/core';
|
||||||
|
|
||||||
|
@Component({
|
||||||
|
selector: 'app-image-converter',
|
||||||
|
templateUrl: 'image-converter.component.html',
|
||||||
|
styleUrls: ['image-converter.component.scss']
|
||||||
|
})
|
||||||
|
export class ImageConverterComponent {
|
||||||
|
|
||||||
|
}
|
||||||
Reference in New Issue
Block a user