Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat: support screenshots #13

Draft
wants to merge 1 commit into
base: main
Choose a base branch
from
Draft
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
118 changes: 117 additions & 1 deletion src/client.ts
Original file line number Diff line number Diff line change
@@ -1,6 +1,11 @@
import type * as prismic from "@prismicio/client";

import type { AbortSignalLike, FetchLike, RequestInitLike } from "./types";
import type {
AbortSignalLike,
FetchLike,
RequestInitLike,
ScreenshotACLResponse,
} from "./types";

import {
BulkTransactionConfirmationError,
@@ -21,6 +26,17 @@
*/
const DEFAULT_CUSTOM_TYPES_API_ENDPOINT = "https://customtypes.prismic.io";

/**
* The default ACL endpoint for the Prismic S3 screenshots bucket.
*/
const DEFAULT_SCREENSHOT_ACL_ENDPOINT =
"https://0yyeb2g040.execute-api.us-east-1.amazonaws.com/prod/";

/**
* The lifetime of a screnshot ACL in milliseconds.
*/
const SCREENSHOT_ACL_TTL = 300_000;

/**
* Configuration for creating a `CustomTypesClient`.
*/
@@ -55,6 +71,12 @@
* overriden on a per-query basis using the query's `fetchOptions` parameter.
*/
fetchOptions?: RequestInitLike;

/**
* The ACL endpoint for the Prismic S3 screenshots bucket. The standard
* endpoint will be used if no value is provided.
*/
screenshotACLEndpoint: string;
};

/**
@@ -98,6 +120,12 @@
deleteDocuments?: boolean;
};

type ScreenshotACL = {
uploadEndpoint: string;
imgixEndpoint: string;
formDataFields: Record<string, string>;
};

/**
* Create a `RequestInit` object for a POST `fetch` request. The provided body
* will be run through `JSON.stringify`.
@@ -113,6 +141,15 @@
};
};

const updateSliceVariationImageInPlace = async <

Check failure on line 144 in src/client.ts

GitHub Actions / ci (ubuntu-latest, 18)

'updateSliceVariationImageInPlace' is assigned a value but never used. Allowed unused vars must match /^_/u
TSharedSliceModelVariation extends prismic.SharedSliceModelVariation,
>(
variation: TSharedSliceModelVariation,
url: URL,
): Promise<TSharedSliceModelVariation> => {
return { ...variation, imageUrl: url.toString() };
};

/**
* Create a client for the Prismic Custom Types API.
*/
@@ -159,6 +196,15 @@
*/
fetchOptions?: RequestInitLike;

/**
* The ACL endpoint for the Prismic S3 screenshots bucket. The standard
* endpoint will be used if no value is provided.
*/
screenshotACLEndpoint: string;

private screenshotACL?: ScreenshotACL;
private screenshotACLExpiresAt?: number;

/**
* Create a client for the Prismic Custom Types API.
*/
@@ -167,6 +213,8 @@
this.endpoint = config.endpoint || DEFAULT_CUSTOM_TYPES_API_ENDPOINT;
this.token = config.token;
this.fetchOptions = config.fetchOptions;
this.screenshotACLEndpoint =
config.screenshotACLEndpoint || DEFAULT_SCREENSHOT_ACL_ENDPOINT;

// TODO: Remove the following `if` statement in v2.
//
@@ -489,6 +537,74 @@
return resolvedOperations;
}

private async uploadImage(data: Blob): Promise<URL> {

Check failure on line 540 in src/client.ts

GitHub Actions / ci (ubuntu-latest, 18)

'data' is defined but never used. Allowed unused args must match /^_/u
const acl = await this.fetchScreenshotACL();

const formData = new FormData();
for (const fieldKey in acl.formDataFields) {
formData.append(fieldKey, acl.formDataFields[fieldKey]);
}

// TODO: Send the image. See `@slicemachine/manager`'s
// `SliceManager.prototype.uploadScreenshot` method.
//
// Since tokens are secret, this package should only be used on
// the server, or in very specific browser-side cases.
//
// We don't necessarily need to use Node.js-only packages, but
// we shouldn't be worried too much about installing larger
// packages.
}

private async fetchScreenshotACL(): Promise<ScreenshotACL> {
if (
this.screenshotACL &&
this.screenshotACLExpiresAt &&
this.screenshotACLExpiresAt < Date.now()
) {
return this.screenshotACL;
}

const url = new URL("./create", this.screenshotACLEndpoint);
const res = await this.fetchFn(url.toString(), {
...this.fetchOptions,
headers: {
Authorization: `Bearer ${this.token}`,
Repository: this.repositoryName,
...this.fetchOptions?.headers,
},
});

let json: ScreenshotACLResponse;
try {
json = await res.json();
} catch (error) {
// Response is not JSON
throw new Error(`Invalid ACL response from ${url}.`, { cause: error });
}

if ("values" in json) {
const acl = {
uploadEndpoint: json.values.url,
imgixEndpoint: json.imgixEndpoint,
formDataFields: json.values.fields,
};

this.screenshotACL = acl;
this.screenshotACLExpiresAt = Date.now() + SCREENSHOT_ACL_TTL;

return acl;
}

if ("error" in json || "message" in json || "Message" in json) {
const errorMessage = json.error || json.message || json.Message;

throw new Error(`Failed to create an AWS ACL: ${errorMessage}`);
}

throw new Error(`Invalid ACL response from ${url}.`);
}

/**
* Performs a network request using the configured `fetch` function. It
* assumes all successful responses will have a JSON content type. It also
14 changes: 14 additions & 0 deletions src/types.ts
Original file line number Diff line number Diff line change
@@ -64,3 +64,17 @@ export interface ResponseLike {
// eslint-disable-next-line @typescript-eslint/no-explicit-any
text(): Promise<any>;
}

export type ScreenshotACLResponse =
| {
values: {
url: string;
fields: Record<string, string>;
};
imgixEndpoint: string;
}
| {
message?: string;
Message?: string;
error?: string;
};
Loading