Provide a public variable "NEXT_PUBLIC_KEYSTATIC_STORAGE_KIND" to allow Keystatic to detect the storage client-side. This is fundamental to make the Admin work.
This commit is contained in:
@@ -1,18 +1,16 @@
|
|||||||
import { z } from 'zod';
|
import { z } from 'zod';
|
||||||
|
|
||||||
|
import { KeystaticStorage } from './keystatic-storage';
|
||||||
import { keyStaticConfig } from './keystatic.config';
|
import { keyStaticConfig } from './keystatic.config';
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The kind of storage to use for the Keystatic reader.
|
* @name createKeystaticReader
|
||||||
*/
|
* @description Creates a new Keystatic reader instance.
|
||||||
const STORAGE_KIND = process.env.KEYSTATIC_STORAGE_KIND ?? 'local';
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Creates a new Keystatic reader instance.
|
|
||||||
*/
|
*/
|
||||||
export async function createKeystaticReader() {
|
export async function createKeystaticReader() {
|
||||||
switch (STORAGE_KIND) {
|
switch (KeystaticStorage.kind) {
|
||||||
case 'local': {
|
case 'local': {
|
||||||
|
// we need to import this dynamically to avoid parsing the package in edge environments
|
||||||
if (process.env.NEXT_RUNTIME === 'nodejs') {
|
if (process.env.NEXT_RUNTIME === 'nodejs') {
|
||||||
const { createReader } = await import('@keystatic/core/reader');
|
const { createReader } = await import('@keystatic/core/reader');
|
||||||
|
|
||||||
@@ -26,28 +24,34 @@ export async function createKeystaticReader() {
|
|||||||
|
|
||||||
case 'github':
|
case 'github':
|
||||||
case 'cloud': {
|
case 'cloud': {
|
||||||
const githubConfig = z
|
|
||||||
.object({
|
|
||||||
token: z.string({
|
|
||||||
description: 'The GitHub token to use for authentication.',
|
|
||||||
}),
|
|
||||||
repo: z.custom<`${string}/${string}`>(),
|
|
||||||
pathPrefix: z.string().optional(),
|
|
||||||
})
|
|
||||||
.parse({
|
|
||||||
token: process.env.KEYSTATIC_GITHUB_TOKEN,
|
|
||||||
repo: process.env.KEYSTATIC_STORAGE_REPO,
|
|
||||||
pathPrefix: process.env.KEYSTATIC_PATH_PREFIX,
|
|
||||||
});
|
|
||||||
|
|
||||||
const { createGitHubReader } = await import(
|
const { createGitHubReader } = await import(
|
||||||
'@keystatic/core/reader/github'
|
'@keystatic/core/reader/github'
|
||||||
);
|
);
|
||||||
|
|
||||||
return createGitHubReader(keyStaticConfig, githubConfig);
|
return createGitHubReader(
|
||||||
|
keyStaticConfig,
|
||||||
|
getKeystaticGithubConfiguration(),
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
default:
|
default:
|
||||||
throw new Error(`Unknown storage kind`);
|
throw new Error(`Unknown storage kind`);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function getKeystaticGithubConfiguration() {
|
||||||
|
return z
|
||||||
|
.object({
|
||||||
|
token: z.string({
|
||||||
|
description:
|
||||||
|
'The GitHub token to use for authentication. Please provide the value through the "KEYSTATIC_GITHUB_TOKEN" environment variable.',
|
||||||
|
}),
|
||||||
|
repo: z.custom<`${string}/${string}`>(),
|
||||||
|
pathPrefix: z.string().optional(),
|
||||||
|
})
|
||||||
|
.parse({
|
||||||
|
token: process.env.KEYSTATIC_GITHUB_TOKEN,
|
||||||
|
repo: process.env.KEYSTATIC_STORAGE_REPO,
|
||||||
|
pathPrefix: process.env.KEYSTATIC_PATH_PREFIX,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|||||||
67
packages/cms/keystatic/src/keystatic-storage.ts
Normal file
67
packages/cms/keystatic/src/keystatic-storage.ts
Normal file
@@ -0,0 +1,67 @@
|
|||||||
|
import { CloudConfig, GitHubConfig, LocalConfig } from '@keystatic/core';
|
||||||
|
import { z } from 'zod';
|
||||||
|
|
||||||
|
type ZodOutputFor<T> = z.ZodType<T, z.ZodTypeDef, unknown>;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @name STORAGE_KIND
|
||||||
|
* @description The kind of storage to use for the Keystatic reader.
|
||||||
|
*
|
||||||
|
* This can be provided through the `KEYSTATIC_STORAGE_KIND` environment variable or 'NEXT_PUBLIC_KEYSTATIC_STORAGE_KIND'.
|
||||||
|
* The previous environment variable `KEYSTATIC_STORAGE_KIND` is deprecated - as Keystatic may need this to be available in the client-side.
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
const STORAGE_KIND = process.env.NEXT_PUBLIC_KEYSTATIC_STORAGE_KIND ??
|
||||||
|
/* @deprecated */
|
||||||
|
process.env.KEYSTATIC_STORAGE_KIND ??
|
||||||
|
'local';
|
||||||
|
|
||||||
|
const PROJECT = process.env.KEYSTATIC_STORAGE_PROJECT;
|
||||||
|
const REPO = process.env.KEYSTATIC_STORAGE_REPO;
|
||||||
|
const BRANCH_PREFIX = process.env.KEYSTATIC_STORAGE_BRANCH_PREFIX;
|
||||||
|
const PATH_PREFIX = process.env.KEYSTATIC_PATH_PREFIX;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @name local
|
||||||
|
* @description The configuration for the local storage.
|
||||||
|
*/
|
||||||
|
const local = z.object({
|
||||||
|
kind: z.literal('local'),
|
||||||
|
}) satisfies ZodOutputFor<LocalConfig['storage']>;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @name cloud
|
||||||
|
* @description The configuration for the cloud storage.
|
||||||
|
*/
|
||||||
|
const cloud = z.object({
|
||||||
|
kind: z.literal('cloud'),
|
||||||
|
project: z.string({
|
||||||
|
description: `The Keystatic Cloud project. Please provide the value through the "KEYSTATIC_STORAGE_PROJECT" environment variable.`,
|
||||||
|
}).min(1),
|
||||||
|
branchPrefix: z.string().optional(),
|
||||||
|
pathPrefix: z.string().optional(),
|
||||||
|
}) satisfies ZodOutputFor<CloudConfig['storage']>;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @name github
|
||||||
|
* @description The configuration for the GitHub storage.
|
||||||
|
*/
|
||||||
|
const github = z.object({
|
||||||
|
kind: z.literal('github'),
|
||||||
|
repo: z.custom<`${string}/${string}`>(),
|
||||||
|
branchPrefix: z.string().optional(),
|
||||||
|
pathPrefix: z.string().optional(),
|
||||||
|
}) satisfies ZodOutputFor<GitHubConfig['storage']>;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @name KeystaticStorage
|
||||||
|
* @description The configuration for the Keystatic storage. This is used to determine where the content is stored.
|
||||||
|
* This configuration is validated through Zod to ensure that the configuration is correct.
|
||||||
|
*/
|
||||||
|
export const KeystaticStorage = z.union([local, cloud, github]).parse({
|
||||||
|
kind: STORAGE_KIND,
|
||||||
|
project: PROJECT,
|
||||||
|
repo: REPO,
|
||||||
|
branchPrefix: BRANCH_PREFIX,
|
||||||
|
pathPrefix: PATH_PREFIX,
|
||||||
|
});
|
||||||
@@ -1,47 +1,7 @@
|
|||||||
import {
|
import { collection, config, fields } from '@keystatic/core';
|
||||||
CloudConfig,
|
|
||||||
GitHubConfig,
|
|
||||||
LocalConfig,
|
|
||||||
collection,
|
|
||||||
config,
|
|
||||||
fields,
|
|
||||||
} from '@keystatic/core';
|
|
||||||
import { Entry } from '@keystatic/core/reader';
|
import { Entry } from '@keystatic/core/reader';
|
||||||
import { z } from 'zod';
|
|
||||||
|
|
||||||
type ZodOutputFor<T> = z.ZodType<T, z.ZodTypeDef, unknown>;
|
import { KeystaticStorage } from './keystatic-storage';
|
||||||
|
|
||||||
const local = z.object({
|
|
||||||
kind: z.literal('local'),
|
|
||||||
}) satisfies ZodOutputFor<LocalConfig['storage']>;
|
|
||||||
|
|
||||||
const cloud = z.object({
|
|
||||||
kind: z.literal('cloud'),
|
|
||||||
project: z.string().min(1),
|
|
||||||
branchPrefix: z.string().optional(),
|
|
||||||
pathPrefix: z.string().optional(),
|
|
||||||
}) satisfies ZodOutputFor<CloudConfig['storage']>;
|
|
||||||
|
|
||||||
const github = z.object({
|
|
||||||
kind: z.literal('github'),
|
|
||||||
repo: z.custom<`${string}/${string}`>(),
|
|
||||||
branchPrefix: z.string().optional(),
|
|
||||||
pathPrefix: z.string().optional(),
|
|
||||||
githubToken: z.string({
|
|
||||||
description:
|
|
||||||
'The GitHub token to use for authentication with the GitHub API',
|
|
||||||
required_error: 'Please provide a GitHub token',
|
|
||||||
}),
|
|
||||||
}) satisfies ZodOutputFor<GitHubConfig['storage']>;
|
|
||||||
|
|
||||||
const storage = z.union([local, cloud, github]).parse({
|
|
||||||
kind: process.env.KEYSTATIC_STORAGE_KIND ?? 'local',
|
|
||||||
project: process.env.KEYSTATIC_STORAGE_PROJECT,
|
|
||||||
repo: process.env.KEYSTATIC_STORAGE_REPO,
|
|
||||||
branchPrefix: process.env.KEYSTATIC_STORAGE_BRANCH_PREFIX,
|
|
||||||
githubToken: process.env.KEYSTATIC_GITHUB_TOKEN,
|
|
||||||
pathPrefix: process.env.KEYSTATIC_PATH_PREFIX,
|
|
||||||
});
|
|
||||||
|
|
||||||
export const keyStaticConfig = createKeyStaticConfig(
|
export const keyStaticConfig = createKeyStaticConfig(
|
||||||
process.env.NEXT_PUBLIC_KEYSTATIC_CONTENT_PATH ?? '',
|
process.env.NEXT_PUBLIC_KEYSTATIC_CONTENT_PATH ?? '',
|
||||||
@@ -85,13 +45,13 @@ function createKeyStaticConfig(path = '') {
|
|||||||
}
|
}
|
||||||
|
|
||||||
const cloud = {
|
const cloud = {
|
||||||
project: storage.kind === 'cloud' ? storage.project : '',
|
project: KeystaticStorage.kind === 'cloud' ? KeystaticStorage.project : '',
|
||||||
};
|
};
|
||||||
|
|
||||||
const collections = getKeystaticCollections(path);
|
const collections = getKeystaticCollections(path);
|
||||||
|
|
||||||
return config({
|
return config({
|
||||||
storage,
|
storage: KeystaticStorage,
|
||||||
cloud,
|
cloud,
|
||||||
collections,
|
collections,
|
||||||
});
|
});
|
||||||
|
|||||||
Reference in New Issue
Block a user