All files / render / server / server.ts

50.00% Branches 9/18
100.00% Lines 46/46
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
 
 
 
 
 
x3
x3
x3
x3
 
x3
 
 
 
 
 
 
 
 
 
x3
 
 
 
 
 
x3
x6
x6
x6
x6
x6
x48
x6
 
x6
x6
x6
 
 
x6
x76
x76
x76
 
x19
x19
 
 
x6
 
 
x6
 
 
 
 
 
 
 
x6
x8
x8
x6
x7
x7
 
 
 
 
 
 
 
 
 
 
 
x6
x22
x88
x110
x22
x22
x22
x22
x220
x22
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
x6
x120
x15
 
 
x3
x3
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
I




















































































I





































































































// Imports
import type { Arg, Directive, Promisable, RendererOptions, RendererRenderOptions } from "@mizu/internal/engine"
import type { CallbackSource, GlobSource, StringSource, URLSource } from "./generate.ts"
// deno-lint-ignore no-external-import
import type { Buffer } from "node:buffer"
import { Context, Renderer } from "@mizu/internal/engine"
import { Window } from "@mizu/internal/vdom"
import defaults from "./defaults.ts"
import { generate } from "./generate.ts"
// deno-lint-ignore no-external-import
import { mkdir, readdir, readFile as read, rm, stat, writeFile as write } from "node:fs/promises"
export type * from "@mizu/internal/engine"
export type { CallbackSource, GlobSource, StringSource, URLSource } from "./generate.ts"

/**
 * Server side renderer.
 *
 * See {@link https://mizu.sh | mizu.sh documentation} for more details.
 * @module
 */
export class Server {
  /**
   * Default options for {@linkcode Server}.
   *
   * These default options are merged with the provided options when creating a new {@linkcode Server} instance.
   */
  static defaults = {
    directives: defaults,
    context: {},
    generate: {
      output: "./output",
      clean: true,
      fs: { stat, read, write, rm, readdir, mkdir },
    },
    // deno-lint-ignore no-console
    warn: console.warn,
    debug: undefined,
  } as unknown as Required<ServerOptions>

  /** {@linkcode Server} constructor. */
  constructor(options?: ServerOptions) {
    this.#options = { ...Server.defaults, ...options }
    this.#options.generate = { ...Server.defaults.generate, ...options?.generate }
    this.#options.generate.fs = { ...Server.defaults.generate.fs, ...options?.generate?.fs }
    // deno-lint-ignore no-explicit-any
    this.#context = new Context<any>(this.#options.context)
  }

  /** Options for {@linkcode Renderer} instantiation. */
  readonly #options

  /** Linked {@linkcode Context}. */
  #context

  /**
   * Default rendering context.
   *
   * All properties assigned to this object are accessible during rendering.
   */
  // deno-lint-ignore no-explicit-any
  get context(): Record<PropertyKey, any> {
    return this.#context.target
  }
  set context(context: Record<PropertyKey, unknown>) {
    this.#context = new Context(context)
  }

  /**
   * Parse an HTML string and render all subtrees.
   *
   * The {@linkcode https://mizu.sh/#mizu | *mizu} attribute is only required if `implicit` is set to `false`.
   *
   * ```ts
   * const mizu = new Server({ context: { foo: "bar" } })
   * await mizu.render(`<html><body><a ~test.text="foo"></a></body></html>`)
   * ```
   */
  async render(content: string | Arg<Renderer["render"]>, options?: ServerRenderOptions & Pick<ServerOptions, "warn">): Promise<string> {
    await using window = new Window(typeof content === "string" ? content : `<body>${content.outerHTML}</body>`)
    const { directives, warn, debug, context: _context } = { ...this.#options, ...options }
    const renderer = await new Renderer(window, { directives, warn, debug }).ready
    let context = this.#context
    if (_context) {
      context = context.with(_context)
    }
    return await renderer.render(renderer.document.documentElement, { implicit: true, ...options, context, state: { $renderer: "server", ...options?.state }, stringify: true })
  }

  /**
   * Generate static files from various sources.
   *
   * Options:
   * - `output`: Specify the path to the output directory.
   * - `clean`: Empty the `output` directory before generating files.
   *
   * Supported sources:
   * - {@linkcode StringSource}: Generate content from raw strings.
   * - {@linkcode GlobSource}: Generate content from local files matching the provided glob patterns.
   * - {@linkcode CallbackSource}: Generate content from callback returns.
   * - {@linkcode URLSource}: Generate content from fetched URLs.
   *
   * Each source can be templated using mizu rendering by passing a `render` option.
   *
   * ```ts
   * const mizu = new Server({ directives: ["@mizu/test"], generate: { output: "/fake/output" } })
   * await mizu.generate(
   *   [
   *     // Copy content from strings
   *     [ "<p>foo</p>", "string.html" ],
   *     [ "<p ~test.text='foo'></p>", "string_render.html", { render: { context: { foo: "bar" } } } ],
   *     // Copy content from local files
   *     [ "**\/*", "public", { directory: "/fake/static" } ],
   *     [ "*.html", "public", { directory: "/fake/partials", render: { context: { foo: "bar "} } } ],
   *     // Copy content from callback return
   *     [ () => JSON.stringify({ foo: "bar" }), "callback.json" ],
   *     [ () => `<p ~test.text="'foo'"></p>`, "callback.html", { render: { context: { foo: "bar" } } } ],
   *     // Copy content from URL
   *     [ new URL(`data:text/html,<p>foobar</p>`), "url.html" ],
   *     [ new URL(`data:text/html,<p ~test.text="foo"></p>`), "url_render.html", { render: { context: { foo: "bar" } } } ],
   *   ],
   *   // No-op: do not actually write files and directories
   *   { fs: { readdir: () => Promise.resolve([] as string[]), mkdir: () => null as any, write: () => null as any } },
   * )
   * ```
   */
  generate(sources: Array<StringSource | GlobSource | CallbackSource | URLSource>, options?: ServerGenerateOptions): Promise<void> {
    return generate(this, sources, { ...this.#options.generate, ...options, fs: { ...this.#options.generate.fs, ...options?.fs } } as Arg<typeof generate, 2>)
  }

  /** Default {@linkcode Server} instance. */
  static readonly default = new Server() as Server
}

/** {@linkcode Server} options. */
export type ServerOptions = Pick<RendererOptions, "warn" | "debug"> & {
  /** Default directives. */
  directives?: Array<Partial<Directive> | string>
  /**
   * Initial rendering {@linkcode Context}.
   *
   * It can be modified later using the {@linkcode Client.context} property.
   */
  context?: ConstructorParameters<typeof Context>[0]
  /** Default {@linkcode Server.generate} options. */
  generate?: ServerGenerateOptions
}

/** {@linkcode Server.render} options. */
export type ServerRenderOptions = Pick<RendererRenderOptions, "implicit" | "select" | "throw"> & {
  /**
   * Rendering context.
   *
   * Values from {@linkcode Client.context} are inherited.
   */
  context?: Arg<Context["with"]>
  /**
   * Initial state.
   *
   * It is populated with `$renderer: "server"` by default.
   */
  state?: Arg<Renderer["render"], 1, true>["state"]
}

/** {@linkcode Server.generate} options. */
export type ServerGenerateOptions = {
  /** Output directory. */
  output?: string
  /** Clean output directory before generation. */
  clean?: boolean
  /** File system options. */
  fs?: Partial<ServerGenerateFileSystemOptions>
}

/** File system options. */
export type ServerGenerateFileSystemOptions = {
  /** Stat callback. */
  stat: (path: string) => Promise<{ isFile: boolean | (() => boolean); isDirectory: boolean | (() => boolean) }>
  /** Read callback. */
  read: (path: string) => Promise<Buffer>
  /** Write callback. */
  write: (path: string, data: Buffer) => Promisable<void>
  /** Remove callback. */
  rm: (path: string, options?: { recursive?: boolean }) => Promisable<unknown>
  /** Make directory callback. */
  mkdir: (path: string, options?: { recursive?: boolean }) => Promisable<unknown>
  /** Read directory callback. */
  readdir: (path: string) => Promise<string[] | { name: string }[]>
}