Refactor and add generators for keystatic and package

The changes include refactoring the create-reader in cms keystatic and changing @keystatic/next version. Additionally, code generators for keystatic and package have been added, and corresponding templates have been organized into their respective directories.
This commit is contained in:
giancarlo
2024-05-07 18:52:30 +07:00
parent 7c447c8848
commit bd79563dcd
14 changed files with 185 additions and 92 deletions

View File

@@ -33,6 +33,7 @@
"dependencies": {
"@edge-csrf/nextjs": "2.0.0",
"@hookform/resolvers": "^3.3.4",
"@keystatic/next": "^5.0.0",
"@kit/accounts": "workspace:^",
"@kit/admin": "workspace:^",
"@kit/auth": "workspace:^",
@@ -42,6 +43,7 @@
"@kit/database-webhooks": "workspace:^",
"@kit/email-templates": "workspace:^",
"@kit/i18n": "workspace:^",
"@kit/keystatic": "workspace:^",
"@kit/mailers": "workspace:^",
"@kit/monitoring": "workspace:^",
"@kit/next": "workspace:^",

View File

@@ -16,7 +16,7 @@
},
"dependencies": {
"@keystatic/core": "0.5.14",
"@keystatic/next": "5.0.0",
"@keystatic/next": "^5.0.0",
"@markdoc/markdoc": "^0.4.0"
},
"devDependencies": {

View File

@@ -9,14 +9,10 @@ export async function createKeystaticReader() {
switch (STORAGE_KIND) {
case 'local': {
if (process.env.NEXT_RUNTIME === 'nodejs') {
const path = await import('node:path');
const { default: config } = await import('./keystatic.config');
const { createReader } = await import('@keystatic/core/reader');
const contentPath = process.env.NEXT_PUBLIC_KEYSTATIC_CONTENT_PATH;
const repositoryPath = path.join(process.cwd(), contentPath as string);
return createReader(repositoryPath, config);
return createReader(process.cwd(), config);
} else {
// we should never get here but the compiler requires the check
// to ensure we don't parse the package at build time

View File

@@ -1,2 +1,3 @@
export * from './keystatic-client';
export * from './content-renderer';
export * from './keystatic.config';

View File

@@ -2,6 +2,6 @@ import { makeRouteHandler } from '@keystatic/next/route-handler';
import config from './keystatic.config';
export const { POST, GET } = makeRouteHandler({
export const keystaticRouteHandlers = makeRouteHandler({
config,
});

View File

@@ -39,7 +39,9 @@ const storage = z.union([local, cloud, github]).parse({
pathPrefix: process.env.KEYSTATIC_PATH_PREFIX,
});
const keyStaticConfig = createKeyStaticConfig();
const keyStaticConfig = createKeyStaticConfig(
process.env.NEXT_PUBLIC_KEYSTATIC_CONTENT_PATH ?? 'content',
);
export default keyStaticConfig;
@@ -75,14 +77,14 @@ export type PostEntryProps = Entry<
(typeof keyStaticConfig)['collections']['posts']
>;
function createKeyStaticConfig() {
function createKeyStaticConfig(path: string) {
return config({
storage,
collections: {
posts: collection({
label: 'Posts',
slugField: 'title',
path: `posts/*`,
path: `${path}/posts/*`,
format: { contentField: 'content' },
schema: {
title: fields.slug({ name: { label: 'Title' } }),
@@ -107,7 +109,7 @@ function createKeyStaticConfig() {
documentation: collection({
label: 'Documentation',
slugField: 'title',
path: `documentation/**`,
path: `${path}/documentation/**`,
format: { contentField: 'content' },
schema: {
title: fields.slug({ name: { label: 'Title' } }),

View File

@@ -1,83 +1,11 @@
import { execSync } from "node:child_process";
import type { PlopTypes } from "@turbo/gen";
import type { PlopTypes } from '@turbo/gen';
import { createKeystaticAdminGenerator } from './templates/keystatic/generator';
import { createPackageGenerator } from './templates/package/generator';
// List of generators to be registered
const generators = [createPackageGenerator, createKeystaticAdminGenerator];
export default function generator(plop: PlopTypes.NodePlopAPI): void {
plop.setGenerator("init", {
description: "Generate a new package for the Monorepo",
prompts: [
{
type: "input",
name: "name",
message:
"What is the name of the package? (You can skip the `@kit/` prefix)",
},
{
type: "input",
name: "deps",
message:
"Enter a space separated list of dependencies you would like to install",
},
],
actions: [
(answers) => {
if ("name" in answers && typeof answers.name === "string") {
if (answers.name.startsWith("@kit/")) {
answers.name = answers.name.replace("@kit/", "");
}
}
return "Config sanitized";
},
{
type: "add",
path: "packages/{{ name }}/package.json",
templateFile: "templates/package.json.hbs",
},
{
type: "add",
path: "packages/{{ name }}/tsconfig.json",
templateFile: "templates/tsconfig.json.hbs",
},
{
type: "add",
path: "packages/{{ name }}/index.ts",
template: "export * from './src';",
},
{
type: "add",
path: "packages/{{ name }}/src/index.ts",
template: "export const name = '{{ name }}';",
},
{
type: "modify",
path: "packages/{{ name }}/package.json",
async transform(content, answers) {
const pkg = JSON.parse(content);
for (const dep of answers.deps.split(" ").filter(Boolean)) {
const version = await fetch(
`https://registry.npmjs.org/-/package/${dep}/dist-tags`,
)
.then((res) => res.json())
.then((json) => json.latest);
pkg.dependencies![dep] = `^${version}`;
}
return JSON.stringify(pkg, null, 2);
},
},
async (answers) => {
/**
* Install deps and format everything
*/
execSync("pnpm manypkg fix", {
stdio: "inherit",
});
execSync(
`pnpm prettier --write packages/${
(answers as { name: string }).name
}/** --list-different`,
);
return "Package scaffolded";
},
],
});
generators.forEach((gen) => gen(plop));
}

View File

@@ -0,0 +1,65 @@
import type { PlopTypes } from '@turbo/gen';
import { execSync } from 'node:child_process';
export function createKeystaticAdminGenerator(plop: PlopTypes.NodePlopAPI) {
return plop.setGenerator('init', {
description: 'Generate a the admin for Keystatic',
prompts: [],
actions: [
{
type: 'add',
path: 'apps/web/app/keystatic/layout.tsx',
templateFile: 'templates/keystatic/layout.tsx.hbs',
},
{
type: 'add',
path: 'apps/web/app/keystatic/[[...params]]/page.tsx',
templateFile: 'templates/keystatic/page.tsx.hbs',
},
{
type: 'add',
path: 'apps/web/app/api/keystatic/[...params]/route.ts',
templateFile: 'templates/keystatic/route.ts.hbs',
},
{
type: 'modify',
path: 'apps/web/package.json',
async transform(content, answers) {
const pkg = JSON.parse(content);
const dep = `@keystatic/next`;
const version = await fetch(
`https://registry.npmjs.org/-/package/${dep}/dist-tags`,
)
.then((res) => res.json())
.then((json) => json.latest);
pkg.dependencies![dep] = `^${version}`;
pkg.dependencies!['@kit/keystatic'] = `workspace:*`;
return JSON.stringify(pkg, null, 2);
},
},
async (answers) => {
/**
* Install deps and format everything
*/
execSync('pnpm manypkg fix', {
stdio: 'inherit',
});
execSync('pnpm i', {
stdio: 'inherit',
});
execSync(
`pnpm prettier --write packages/${
(answers as { name: string }).name
}/** --list-different`,
);
return `Keystatic admin generated!`;
},
],
});
}

View File

@@ -0,0 +1,7 @@
import KeystaticAdmin from '@kit/keystatic/admin';
export default function Layout() {
return (
<KeystaticAdmin />
);
}

View File

@@ -0,0 +1,5 @@
// src/app/keystatic/[[...params]]/page.tsx
export default function Page() {
return null;
}

View File

@@ -0,0 +1,3 @@
import { keystaticRouteHandlers } from '@kit/keystatic/route-handler';
export const { POST, GET } = keystaticRouteHandlers;

View File

@@ -0,0 +1,84 @@
import type { PlopTypes } from '@turbo/gen';
import { execSync } from 'node:child_process';
export function createPackageGenerator(plop: PlopTypes.NodePlopAPI) {
plop.setGenerator('init', {
description: 'Generate a new package for the Monorepo',
prompts: [
{
type: 'input',
name: 'name',
message:
'What is the name of the package? (You can skip the `@kit/` prefix)',
},
{
type: 'input',
name: 'deps',
message:
'Enter a space separated list of dependencies you would like to install',
},
],
actions: [
(answers) => {
if ('name' in answers && typeof answers.name === 'string') {
if (answers.name.startsWith('@kit/')) {
answers.name = answers.name.replace('@kit/', '');
}
}
return 'Config sanitized';
},
{
type: 'add',
path: 'packages/{{ name }}/package.json',
templateFile: 'templates/package/package.json.hbs',
},
{
type: 'add',
path: 'packages/{{ name }}/tsconfig.json',
templateFile: 'templates/package/tsconfig.json.hbs',
},
{
type: 'add',
path: 'packages/{{ name }}/index.ts',
template: "export * from './src';",
},
{
type: 'add',
path: 'packages/{{ name }}/src/index.ts',
template: "export const name = '{{ name }}';",
},
{
type: 'modify',
path: 'packages/{{ name }}/package.json',
async transform(content, answers) {
const pkg = JSON.parse(content);
for (const dep of answers.deps.split(' ').filter(Boolean)) {
const version = await fetch(
`https://registry.npmjs.org/-/package/${dep}/dist-tags`,
)
.then((res) => res.json())
.then((json) => json.latest);
pkg.dependencies![dep] = `^${version}`;
}
return JSON.stringify(pkg, null, 2);
},
},
async (answers) => {
/**
* Install deps and format everything
*/
execSync('pnpm manypkg fix', {
stdio: 'inherit',
});
execSync(
`pnpm prettier --write packages/${
(answers as { name: string }).name
}/** --list-different`,
);
return 'Package scaffolded';
},
],
});
}