feat: harden deployment and public api handoff
This commit is contained in:
34
app/api/v1/assets/[id]/download/route.ts
Normal file
34
app/api/v1/assets/[id]/download/route.ts
Normal file
@@ -0,0 +1,34 @@
|
||||
import { jsonError } from "@/lib/server/api";
|
||||
import { authenticatePublicApiRequest } from "@/lib/server/public-api-auth";
|
||||
import { getPublicApiAsset } from "@/lib/server/public-api-assets";
|
||||
import { publicApiError } from "@/lib/server/public-api-response";
|
||||
import { readAssetForDownload } from "@/lib/server/storage";
|
||||
|
||||
export const runtime = "nodejs";
|
||||
|
||||
export async function GET(request: Request, context: { params: Promise<{ id: string }> }) {
|
||||
try {
|
||||
const client = authenticatePublicApiRequest(request);
|
||||
const { id } = await context.params;
|
||||
const asset = await getPublicApiAsset(client.id, id);
|
||||
if (!asset) return jsonError("Asset not found.", 404);
|
||||
const file = await readAssetForDownload(asset);
|
||||
if (!file) return jsonError("Asset file is not downloadable.", 404);
|
||||
return new Response(new Uint8Array(file.bytes), {
|
||||
headers: {
|
||||
"Content-Type": file.contentType,
|
||||
"Content-Length": String(file.bytes.length),
|
||||
"Content-Disposition": contentDisposition(asset.name),
|
||||
"Cache-Control": "private, no-store"
|
||||
}
|
||||
});
|
||||
} catch (error) {
|
||||
return publicApiError(error);
|
||||
}
|
||||
}
|
||||
|
||||
function contentDisposition(fileName: string): string {
|
||||
const clean = fileName.replace(/[\r\n/\\]/g, "_").trim() || "download";
|
||||
const ascii = clean.replace(/[^\x20-\x7E]/g, "_").replace(/"/g, "_");
|
||||
return `attachment; filename="${ascii}"; filename*=UTF-8''${encodeURIComponent(clean)}`;
|
||||
}
|
||||
18
app/api/v1/assets/[id]/route.ts
Normal file
18
app/api/v1/assets/[id]/route.ts
Normal file
@@ -0,0 +1,18 @@
|
||||
import { jsonError, jsonOk } from "@/lib/server/api";
|
||||
import { authenticatePublicApiRequest } from "@/lib/server/public-api-auth";
|
||||
import { getPublicApiAsset } from "@/lib/server/public-api-assets";
|
||||
import { publicApiError } from "@/lib/server/public-api-response";
|
||||
|
||||
export const runtime = "nodejs";
|
||||
|
||||
export async function GET(request: Request, context: { params: Promise<{ id: string }> }) {
|
||||
try {
|
||||
const client = authenticatePublicApiRequest(request);
|
||||
const { id } = await context.params;
|
||||
const asset = await getPublicApiAsset(client.id, id);
|
||||
if (!asset) return jsonError("Asset not found.", 404);
|
||||
return jsonOk({ asset });
|
||||
} catch (error) {
|
||||
return publicApiError(error);
|
||||
}
|
||||
}
|
||||
@@ -1,6 +1,7 @@
|
||||
import { createAsset, listAssets } from "@/lib/server/data-store";
|
||||
import { createAsset } from "@/lib/server/data-store";
|
||||
import { jsonOk, readJsonBody } from "@/lib/server/api";
|
||||
import { authenticatePublicApiRequest } from "@/lib/server/public-api-auth";
|
||||
import { listPublicApiAssets } from "@/lib/server/public-api-assets";
|
||||
import { publicApiError } from "@/lib/server/public-api-response";
|
||||
import { DEFAULT_OWNER_ID, requestOrigin } from "@/lib/server/runtime";
|
||||
import { saveUploadAsset } from "@/lib/server/storage";
|
||||
@@ -10,8 +11,8 @@ export const runtime = "nodejs";
|
||||
|
||||
export async function GET(request: Request) {
|
||||
try {
|
||||
authenticatePublicApiRequest(request);
|
||||
return jsonOk({ assets: await listAssets(DEFAULT_OWNER_ID) });
|
||||
const client = authenticatePublicApiRequest(request);
|
||||
return jsonOk({ assets: await listPublicApiAssets(client.id) });
|
||||
} catch (error) {
|
||||
return publicApiError(error);
|
||||
}
|
||||
|
||||
Reference in New Issue
Block a user