@gradial/aci 0.1.2 → 0.1.3

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/README.md CHANGED
@@ -106,3 +106,22 @@ npm publish --access public
106
106
 
107
107
  The package includes a `prepack` script that builds `dist` before packing or
108
108
  publishing.
109
+
110
+ ## Verify the Published Shape Locally
111
+
112
+ Before publishing, inspect the exact tarball npm will upload. This catches
113
+ package `files` mistakes where local source imports work from the repo but fail
114
+ after npm filters the packed files.
115
+
116
+ From `packages/sdk`:
117
+
118
+ ```bash
119
+ tmpdir="$(mktemp -d /tmp/gradial-aci-pack.XXXXXX)"
120
+ filename="$(npm pack --pack-destination "$tmpdir" | tail -n 1)"
121
+ tar -tzf "$tmpdir/$filename" | sort | rg '^(package/src/cli/|package/src/types/|package/dist/types/)'
122
+ echo "tarball=$tmpdir/$filename"
123
+ ```
124
+
125
+ If a shipped source CLI imports another source file, make sure both paths appear
126
+ in the tarball. For example, `src/cli/compile-registry.mjs` imports
127
+ `src/types/image.ts`, so both files must be included by `package.json#files`.
package/bin/aci CHANGED
Binary file
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@gradial/aci",
3
- "version": "0.1.2",
3
+ "version": "0.1.3",
4
4
  "type": "module",
5
5
  "main": "./dist/index.js",
6
6
  "types": "./dist/index.d.ts",
@@ -9,6 +9,7 @@
9
9
  "bin/aci.js",
10
10
  "bin/aci",
11
11
  "src/cli/*.mjs",
12
+ "src/types/*.ts",
12
13
  "README.md"
13
14
  ],
14
15
  "bin": {
@@ -0,0 +1,59 @@
1
+ import type { IslandMode, VaryDimension } from './render-mode.js';
2
+ import type { ImageSlotContract } from './image.js';
3
+ import type { z } from 'zod';
4
+
5
+ export interface ComponentRenderModes {
6
+ canStatic: boolean;
7
+ canSSR: boolean;
8
+ canClientIsland: boolean;
9
+ }
10
+
11
+ /**
12
+ * A CMS-registered component — sync or async (server components).
13
+ * Accepts content props derived from the Zod schema.
14
+ */
15
+ type CmsComponentFn<TProps = any> =
16
+ | ((props: TProps) => import('react').ReactElement | null)
17
+ | ((props: TProps) => Promise<import('react').ReactElement | null>);
18
+
19
+ /**
20
+ * Infers the props type from a schema.
21
+ * If TSchema is a Zod type, returns z.infer<TSchema>.
22
+ * Otherwise returns Record<string, unknown>.
23
+ */
24
+ type InferContentProps<TSchema> = TSchema extends z.ZodType<infer T> ? T : Record<string, unknown>;
25
+
26
+ export interface ComponentDefinition<TSchema = unknown> {
27
+ name: string;
28
+ /**
29
+ * The React component that renders this CMS block.
30
+ * Accepts at least the schema-derived props, but may accept additional
31
+ * composition params (className, etc.) passed by parent sections.
32
+ */
33
+ component?: CmsComponentFn<any>;
34
+ schema: TSchema;
35
+ renderModes?: ComponentRenderModes;
36
+ render?: ComponentRenderModes;
37
+ defaultIslandMode?: IslandMode;
38
+ varyDimensions?: VaryDimension[];
39
+ vary?: VaryDimension[];
40
+ imageSlots?: Record<string, ImageSlotContract>;
41
+ }
42
+
43
+ export interface ComponentContract<TSchema = unknown> {
44
+ name: string;
45
+ component?: CmsComponentFn<any>;
46
+ schema: TSchema;
47
+ renderModes: ComponentRenderModes;
48
+ render: ComponentRenderModes;
49
+ defaultIslandMode?: IslandMode;
50
+ varyDimensions?: VaryDimension[];
51
+ vary?: VaryDimension[];
52
+ imageSlots?: Record<string, ImageSlotContract>;
53
+ }
54
+
55
+ export type InferComponentProps<TContract> = TContract extends ComponentContract<infer TSchema>
56
+ ? InferContentProps<TSchema>
57
+ : Record<string, unknown>;
58
+
59
+ export type { CmsComponentFn, InferContentProps };
@@ -0,0 +1,36 @@
1
+ import type { RenderCapabilities } from './renderer.js';
2
+
3
+ export interface BareMetalConfig {
4
+ version: '1';
5
+ siteId: string;
6
+ framework: 'astro' | 'next' | 'sveltekit' | 'custom';
7
+ source: {
8
+ root: string;
9
+ outDir?: string;
10
+ };
11
+ componentRegistry: string;
12
+ layoutRegistry: string;
13
+ rendererEntry: string;
14
+ capabilities: RenderCapabilities;
15
+ routes: RouteConfig;
16
+ dam?: DAMConfig;
17
+ externalDependencies?: ExternalDependency[];
18
+ rendererProtocol: 'http' | 'stdio-json';
19
+ }
20
+
21
+ export interface RouteConfig {
22
+ cmsManaged: string;
23
+ frameworkOwned?: string[];
24
+ }
25
+
26
+ export interface ExternalDependency {
27
+ name: string;
28
+ host: string;
29
+ kind?: 'api' | 'client-script';
30
+ description?: string;
31
+ cacheTTL?: number;
32
+ }
33
+
34
+ export interface DAMConfig {
35
+ autoApproveUploads?: boolean;
36
+ }
@@ -0,0 +1,100 @@
1
+ import { z } from 'zod';
2
+
3
+ export interface ImageSource {
4
+ src: string;
5
+ width: number;
6
+ height: number;
7
+ type: string;
8
+ }
9
+
10
+ export interface PictureSource {
11
+ media?: string;
12
+ type: string;
13
+ srcset: string;
14
+ sizes?: string;
15
+ }
16
+
17
+ export interface GradialImage {
18
+ $type: 'gradial.image';
19
+ assetId: string;
20
+ versionId: string;
21
+ alt: string;
22
+ fallback: ImageSource;
23
+ sources: PictureSource[];
24
+ }
25
+
26
+ export interface ImageSlotContract {
27
+ outputs: SlotOutput[];
28
+ formats: string[];
29
+ sizes: string;
30
+ }
31
+
32
+ export interface SlotOutput {
33
+ aspectRatio: string;
34
+ widths: number[];
35
+ media?: string;
36
+ }
37
+
38
+ export const GradialImageSchema = z.object({
39
+ $type: z.literal('gradial.image'),
40
+ assetId: z.string().min(1),
41
+ versionId: z.string().min(1),
42
+ alt: z.string(),
43
+ fallback: z.object({
44
+ src: z.string().min(1),
45
+ width: z.number().int().nonnegative(),
46
+ height: z.number().int().nonnegative(),
47
+ type: z.string().min(1),
48
+ }),
49
+ sources: z.array(z.object({
50
+ media: z.string().optional(),
51
+ type: z.string().min(1),
52
+ srcset: z.string().min(1),
53
+ sizes: z.string().optional(),
54
+ })).nullable().default([]),
55
+ });
56
+
57
+ export type ImageHTMLAttributes = Record<string, string | number | boolean | undefined | null>;
58
+
59
+ export function renderImageHTML(image: GradialImage, attrs: ImageHTMLAttributes = {}): string {
60
+ const imgAttrs = renderAttrs({
61
+ ...attrs,
62
+ src: image.fallback.src,
63
+ alt: image.alt,
64
+ width: image.fallback.width > 0 ? image.fallback.width : undefined,
65
+ height: image.fallback.height > 0 ? image.fallback.height : undefined,
66
+ });
67
+ const img = `<img${imgAttrs}>`;
68
+ if (!image.sources.length) {
69
+ return img;
70
+ }
71
+ const sources = image.sources.map((source) => `<source${renderAttrs({
72
+ media: source.media,
73
+ type: source.type,
74
+ srcset: source.srcset,
75
+ sizes: source.sizes,
76
+ })}>`).join('');
77
+ return `<picture>${sources}${img}</picture>`;
78
+ }
79
+
80
+ function renderAttrs(attrs: ImageHTMLAttributes): string {
81
+ const rendered = Object.entries(attrs)
82
+ .filter(([, value]) => value !== undefined && value !== null && value !== false)
83
+ .map(([name, value]) => value === true ? escapeName(name) : `${escapeName(name)}="${escapeAttr(String(value))}"`);
84
+ if (!rendered.length) {
85
+ return '';
86
+ }
87
+ return ' ' + rendered.join(' ');
88
+ }
89
+
90
+ function escapeName(name: string): string {
91
+ return name.replace(/[^A-Za-z0-9_:-]/g, '');
92
+ }
93
+
94
+ function escapeAttr(value: string): string {
95
+ return value
96
+ .replace(/&/g, '&amp;')
97
+ .replace(/"/g, '&quot;')
98
+ .replace(/</g, '&lt;')
99
+ .replace(/>/g, '&gt;');
100
+ }
@@ -0,0 +1,9 @@
1
+ export * from './config.js';
2
+ export * from './component.js';
3
+ export * from './layout.js';
4
+ export * from './page.js';
5
+ export * from './renderer.js';
6
+ export * from './render-mode.js';
7
+ export * from './image.js';
8
+ export * from './video.js';
9
+ export * from './media.js';
@@ -0,0 +1,29 @@
1
+ import type { FragmentRefNode } from './page.js';
2
+
3
+ export interface LayoutSlot {
4
+ name: string;
5
+ required: boolean;
6
+ }
7
+
8
+ /** Default content for non-required layout slots, keyed by slot name. */
9
+ export type LayoutDefaults = Record<string, FragmentRefNode[]>;
10
+
11
+ export interface LayoutDefinition {
12
+ name: string;
13
+ slots: LayoutSlot[];
14
+ /** Default fragment references for slots not provided by the page. */
15
+ defaults?: LayoutDefaults;
16
+ }
17
+
18
+ export interface LayoutContract {
19
+ name: string;
20
+ slots: LayoutSlot[];
21
+ defaults?: LayoutDefaults;
22
+ }
23
+
24
+ /** A compiled layout file as stored on disk. */
25
+ export interface CompiledLayout {
26
+ name: string;
27
+ slots: LayoutSlot[];
28
+ defaults: LayoutDefaults;
29
+ }
@@ -0,0 +1,125 @@
1
+ import { z } from 'zod';
2
+ import { GradialImageSchema, renderImageHTML, type GradialImage } from './image.js';
3
+ import { GradialVideoSchema, type GradialVideo } from './video.js';
4
+
5
+ // ---------------------------------------------------------------------------
6
+ // GradialAsset — union of all DAM-backed asset types
7
+ //
8
+ // Content authors write: { "$type": "dam.assetRef", "assetId": "..." }
9
+ // The compiler resolves this to a GradialImage or GradialVideo based on
10
+ // the asset's media type in the DAM.
11
+ // ---------------------------------------------------------------------------
12
+
13
+ /** Discriminated union of all compiler-resolved asset types. */
14
+ export type GradialAsset = GradialImage | GradialVideo;
15
+
16
+ // ---------------------------------------------------------------------------
17
+ // Zod schema — discriminated on $type
18
+ // ---------------------------------------------------------------------------
19
+
20
+ export const GradialAssetSchema = z.discriminatedUnion('$type', [
21
+ GradialImageSchema,
22
+ GradialVideoSchema,
23
+ ]);
24
+
25
+ // ---------------------------------------------------------------------------
26
+ // Type guards
27
+ // ---------------------------------------------------------------------------
28
+
29
+ export function isGradialImage(value: unknown): value is GradialImage {
30
+ return (
31
+ typeof value === 'object' &&
32
+ value !== null &&
33
+ (value as Record<string, unknown>).$type === 'gradial.image'
34
+ );
35
+ }
36
+
37
+ export function isGradialVideo(value: unknown): value is GradialVideo {
38
+ return (
39
+ typeof value === 'object' &&
40
+ value !== null &&
41
+ (value as Record<string, unknown>).$type === 'gradial.video'
42
+ );
43
+ }
44
+
45
+ export function isGradialAsset(value: unknown): value is GradialAsset {
46
+ return isGradialImage(value) || isGradialVideo(value);
47
+ }
48
+
49
+ // ---------------------------------------------------------------------------
50
+ // HTML string renderers (framework-agnostic)
51
+ // ---------------------------------------------------------------------------
52
+
53
+ /**
54
+ * Renders a GradialVideo to an HTML string.
55
+ * Uses poster.fallback.src for the poster attribute.
56
+ * Includes <source> elements for format variants.
57
+ */
58
+ export function renderVideoHTML(
59
+ video: GradialVideo,
60
+ attrs: Record<string, string | number | boolean | undefined | null> = {},
61
+ ): string {
62
+ const posterSrc = video.poster?.fallback.src;
63
+ const attrPairs: Array<[string, string | number | boolean | undefined | null]> = [
64
+ ...Object.entries(attrs),
65
+ ['src', video.sources?.length ? undefined : video.src],
66
+ ['poster', posterSrc],
67
+ ['width', video.width > 0 ? video.width : undefined],
68
+ ['height', video.height > 0 ? video.height : undefined],
69
+ ];
70
+ const attrString = renderAttrs(attrPairs);
71
+
72
+ const sources = (video.sources ?? [])
73
+ .map((s) => `<source${renderAttrs([['src', s.src], ['type', s.type]])}>`)
74
+ .join('');
75
+
76
+ // If no format variants, src is on the <video> element itself
77
+ if (!sources) {
78
+ return `<video${attrString}></video>`;
79
+ }
80
+ return `<video${attrString}>${sources}</video>`;
81
+ }
82
+
83
+ /**
84
+ * Renders any GradialAsset to an HTML string.
85
+ * Delegates to renderImageHTML or renderVideoHTML based on $type.
86
+ */
87
+ export function renderAssetHTML(
88
+ asset: GradialAsset,
89
+ attrs: Record<string, string | number | boolean | undefined | null> = {},
90
+ ): string {
91
+ if (asset.$type === 'gradial.image') {
92
+ return renderImageHTML(asset, attrs);
93
+ }
94
+ return renderVideoHTML(asset, attrs);
95
+ }
96
+
97
+ // ---------------------------------------------------------------------------
98
+ // Internal helpers
99
+ // ---------------------------------------------------------------------------
100
+
101
+ function renderAttrs(
102
+ pairs: Array<[string, string | number | boolean | undefined | null]>,
103
+ ): string {
104
+ const rendered = pairs
105
+ .filter(([, value]) => value !== undefined && value !== null && value !== false)
106
+ .map(([name, value]) =>
107
+ value === true
108
+ ? escapeName(name)
109
+ : `${escapeName(name)}="${escapeAttr(String(value))}"`,
110
+ );
111
+ if (!rendered.length) return '';
112
+ return ' ' + rendered.join(' ');
113
+ }
114
+
115
+ function escapeName(name: string): string {
116
+ return name.replace(/[^A-Za-z0-9_:-]/g, '');
117
+ }
118
+
119
+ function escapeAttr(value: string): string {
120
+ return value
121
+ .replace(/&/g, '&amp;')
122
+ .replace(/"/g, '&quot;')
123
+ .replace(/</g, '&lt;')
124
+ .replace(/>/g, '&gt;');
125
+ }
@@ -0,0 +1,48 @@
1
+ import type { RenderMode, VaryDimension } from './render-mode.js';
2
+
3
+ export interface PageDocument {
4
+ path: string;
5
+ layout: string;
6
+ locale: string;
7
+ metadata: PageMetadata;
8
+ regions: Record<string, RegionNode>;
9
+ renderMode: RenderMode;
10
+ }
11
+
12
+ export interface PageMetadata {
13
+ title: string;
14
+ description?: string;
15
+ openGraph?: Record<string, string>;
16
+ canonicalUrl?: string;
17
+ alternateLocales?: { locale: string; path: string }[];
18
+ customHead?: string[];
19
+ }
20
+
21
+ export interface RegionNode {
22
+ kind: 'region';
23
+ name: string;
24
+ children: Array<BlockNode | FragmentRefNode>;
25
+ }
26
+
27
+ export interface BlockNode {
28
+ kind: 'block';
29
+ id: string;
30
+ component: string;
31
+ props: Record<string, unknown>;
32
+ renderHints: BlockRenderHints;
33
+ contentRef?: string;
34
+ }
35
+
36
+ export interface BlockRenderHints {
37
+ canStatic: boolean;
38
+ needsSSR: boolean;
39
+ isIsland: boolean;
40
+ islandMode?: 'ssr' | 'client';
41
+ varyDimensions?: VaryDimension[];
42
+ }
43
+
44
+ export interface FragmentRefNode {
45
+ kind: 'fragment-ref';
46
+ fragmentId: string;
47
+ inline: boolean;
48
+ }
@@ -0,0 +1,18 @@
1
+ export type RenderMode =
2
+ | 'static'
3
+ | 'static-with-fragments'
4
+ | 'prerender-on-demand'
5
+ | 'ssr-page'
6
+ | 'ssr-island'
7
+ | 'client-island';
8
+
9
+ export type IslandMode = 'ssr' | 'client';
10
+
11
+ export type VaryDimension =
12
+ | 'geo:country'
13
+ | 'geo:region'
14
+ | 'geo:city'
15
+ | `cookie:${string}`
16
+ | `header:${string}`
17
+ | 'locale'
18
+ | `query:${string}`;
@@ -0,0 +1,83 @@
1
+ import type { PageDocument } from './page.js';
2
+ import type { VaryDimension } from './render-mode.js';
3
+
4
+ export interface RenderCapabilities {
5
+ staticRender: boolean;
6
+ ssr: boolean;
7
+ ssrIslands: boolean;
8
+ clientIslands: boolean;
9
+ fragmentRender: boolean;
10
+ }
11
+
12
+ export interface GradialRenderer {
13
+ renderPage(request: RenderPageRequest): Promise<RenderPageResult>;
14
+ renderFragment?(request: RenderFragmentRequest): Promise<RenderResult>;
15
+ renderIsland?(request: RenderIslandRequest): Promise<RenderResult>;
16
+ }
17
+
18
+ export interface RenderPageRequest {
19
+ page: PageDocument;
20
+ requestContext?: RequestContext;
21
+ release: ReleaseContext;
22
+ }
23
+
24
+ export interface RequestContext {
25
+ url: string;
26
+ method: string;
27
+ headers: Record<string, string>;
28
+ cookies: Record<string, string>;
29
+ geo?: GeoContext;
30
+ }
31
+
32
+ export interface GeoContext {
33
+ country?: string;
34
+ region?: string;
35
+ city?: string;
36
+ }
37
+
38
+ export interface ReleaseContext {
39
+ releaseId: string;
40
+ codeDigest: string;
41
+ contentSnapshotId: string;
42
+ assetUrl(logicalPath: string): string;
43
+ }
44
+
45
+ export interface RenderFragmentRequest {
46
+ fragmentId: string;
47
+ content: unknown;
48
+ release: ReleaseContext;
49
+ }
50
+
51
+ export interface RenderIslandRequest {
52
+ islandId: string;
53
+ component: string;
54
+ props: Record<string, unknown>;
55
+ requestContext: RequestContext;
56
+ release: ReleaseContext;
57
+ }
58
+
59
+ export interface RenderResult {
60
+ html: string;
61
+ status?: number;
62
+ headers?: Record<string, string>;
63
+ cachePolicy?: CachePolicy;
64
+ }
65
+
66
+ export interface RenderPageResult extends RenderResult {
67
+ islands?: IslandDescriptor[];
68
+ }
69
+
70
+ export interface IslandDescriptor {
71
+ islandId: string;
72
+ component: string;
73
+ props: Record<string, unknown>;
74
+ mode: 'ssr' | 'client';
75
+ varyDimensions?: VaryDimension[];
76
+ }
77
+
78
+ export interface CachePolicy {
79
+ scope: 'public' | 'private' | 'no-store';
80
+ ttl?: number;
81
+ staleWhileRevalidate?: number;
82
+ tags?: string[];
83
+ }
@@ -0,0 +1,66 @@
1
+ import { z } from 'zod';
2
+ import { GradialImageSchema, type GradialImage } from './image.js';
3
+
4
+ // ---------------------------------------------------------------------------
5
+ // Video source variant (format alternatives)
6
+ // ---------------------------------------------------------------------------
7
+
8
+ export interface VideoSource {
9
+ src: string;
10
+ type: string; // MIME type: video/mp4, video/webm
11
+ }
12
+
13
+ // ---------------------------------------------------------------------------
14
+ // GradialVideo — DAM-backed video asset with compiler-resolved derivatives
15
+ // ---------------------------------------------------------------------------
16
+
17
+ export interface GradialVideo {
18
+ /** Discriminator — always 'gradial.video'. */
19
+ $type: 'gradial.video';
20
+ /** DAM asset identifier. */
21
+ assetId: string;
22
+ /** DAM version identifier. */
23
+ versionId: string;
24
+ /** Accessible description of the video content. */
25
+ alt: string;
26
+ /** Primary video URL (compiler-resolved). */
27
+ src: string;
28
+ /** Primary video MIME type (e.g. 'video/mp4'). */
29
+ type: string;
30
+ /** Intrinsic width in pixels. */
31
+ width: number;
32
+ /** Intrinsic height in pixels. */
33
+ height: number;
34
+ /**
35
+ * Poster frame — a full GradialImage with responsive srcset.
36
+ * The compiler extracts or generates this from the video asset.
37
+ */
38
+ poster?: GradialImage;
39
+ /** Format variants (e.g. mp4 + webm). Maps to <video><source> elements. */
40
+ sources?: VideoSource[];
41
+ /** Duration in seconds (if known). */
42
+ duration?: number;
43
+ }
44
+
45
+ // ---------------------------------------------------------------------------
46
+ // Zod schema
47
+ // ---------------------------------------------------------------------------
48
+
49
+ export const VideoSourceSchema = z.object({
50
+ src: z.string().min(1),
51
+ type: z.string().min(1),
52
+ });
53
+
54
+ export const GradialVideoSchema = z.object({
55
+ $type: z.literal('gradial.video'),
56
+ assetId: z.string().min(1),
57
+ versionId: z.string().min(1),
58
+ alt: z.string(),
59
+ src: z.string().min(1),
60
+ type: z.string().min(1),
61
+ width: z.number().int().nonnegative(),
62
+ height: z.number().int().nonnegative(),
63
+ poster: GradialImageSchema.optional(),
64
+ sources: z.array(VideoSourceSchema).optional(),
65
+ duration: z.number().nonnegative().optional(),
66
+ });