docula 0.3.6 → 0.4.0

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.
Files changed (152) hide show
  1. package/README.md +108 -39
  2. package/bin/docula.mjs +8 -0
  3. package/dist/builder.d.ts +34 -0
  4. package/dist/builder.js +182 -0
  5. package/dist/console.d.ts +22 -0
  6. package/dist/console.js +120 -0
  7. package/dist/docula.d.ts +22 -21
  8. package/dist/docula.js +126 -165
  9. package/dist/github.d.ts +22 -0
  10. package/dist/github.js +74 -0
  11. package/dist/helpers/markdown.d.ts +10 -0
  12. package/dist/helpers/markdown.js +24 -0
  13. package/dist/helpers.d.ts +7 -0
  14. package/dist/helpers.js +58 -0
  15. package/dist/options.d.ts +12 -0
  16. package/dist/options.js +47 -0
  17. package/init/docula.config.cjs +10 -0
  18. package/init/favicon.ico +0 -0
  19. package/init/logo.png +0 -0
  20. package/init/variables.css +30 -0
  21. package/package.json +63 -95
  22. package/{init/_includes/assets/css/styles → template/css}/home.css +20 -18
  23. package/{init/_includes/assets/css/styles → template/css}/index.css +7 -0
  24. package/template/includes/footer.hbs +10 -0
  25. package/template/includes/header.hbs +17 -0
  26. package/template/includes/landing/content.hbs +63 -0
  27. package/template/includes/landing/hero.hbs +3 -0
  28. package/template/includes/scripts.hbs +4 -0
  29. package/template/index.hbs +16 -0
  30. package/template/releases.hbs +45 -0
  31. package/bin/docula.js +0 -7
  32. package/dist/config.d.ts +0 -21
  33. package/dist/config.d.ts.map +0 -1
  34. package/dist/config.js +0 -78
  35. package/dist/config.js.map +0 -1
  36. package/dist/docula-plugin.d.ts +0 -9
  37. package/dist/docula-plugin.d.ts.map +0 -1
  38. package/dist/docula-plugin.js +0 -2
  39. package/dist/docula-plugin.js.map +0 -1
  40. package/dist/docula.d.ts.map +0 -1
  41. package/dist/docula.js.map +0 -1
  42. package/dist/eleventy/filters.d.ts +0 -2
  43. package/dist/eleventy/filters.d.ts.map +0 -1
  44. package/dist/eleventy/filters.js +0 -10
  45. package/dist/eleventy/filters.js.map +0 -1
  46. package/dist/eleventy/global-data.d.ts +0 -3
  47. package/dist/eleventy/global-data.d.ts.map +0 -1
  48. package/dist/eleventy/global-data.js +0 -2
  49. package/dist/eleventy/global-data.js.map +0 -1
  50. package/dist/eleventy/shortcodes.d.ts +0 -4
  51. package/dist/eleventy/shortcodes.d.ts.map +0 -1
  52. package/dist/eleventy/shortcodes.js +0 -18
  53. package/dist/eleventy/shortcodes.js.map +0 -1
  54. package/dist/eleventy.d.ts +0 -24
  55. package/dist/eleventy.d.ts.map +0 -1
  56. package/dist/eleventy.js +0 -82
  57. package/dist/eleventy.js.map +0 -1
  58. package/dist/index.d.ts +0 -9
  59. package/dist/index.d.ts.map +0 -1
  60. package/dist/index.js +0 -53
  61. package/dist/index.js.map +0 -1
  62. package/dist/logger.d.ts +0 -4
  63. package/dist/logger.d.ts.map +0 -1
  64. package/dist/logger.js +0 -20
  65. package/dist/logger.js.map +0 -1
  66. package/dist/plugins/algolia.d.ts +0 -21
  67. package/dist/plugins/algolia.d.ts.map +0 -1
  68. package/dist/plugins/algolia.js +0 -54
  69. package/dist/plugins/algolia.js.map +0 -1
  70. package/dist/plugins/github.d.ts +0 -16
  71. package/dist/plugins/github.d.ts.map +0 -1
  72. package/dist/plugins/github.js +0 -68
  73. package/dist/plugins/github.js.map +0 -1
  74. package/dist/plugins/index.d.ts +0 -18
  75. package/dist/plugins/index.d.ts.map +0 -1
  76. package/dist/plugins/index.js +0 -16
  77. package/dist/plugins/index.js.map +0 -1
  78. package/dist/plugins/npm.d.ts +0 -14
  79. package/dist/plugins/npm.d.ts.map +0 -1
  80. package/dist/plugins/npm.js +0 -36
  81. package/dist/plugins/npm.js.map +0 -1
  82. package/dist/plugins/pagefind.d.ts +0 -10
  83. package/dist/plugins/pagefind.d.ts.map +0 -1
  84. package/dist/plugins/pagefind.js +0 -22
  85. package/dist/plugins/pagefind.js.map +0 -1
  86. package/dist/plugins/robots.d.ts +0 -14
  87. package/dist/plugins/robots.d.ts.map +0 -1
  88. package/dist/plugins/robots.js +0 -34
  89. package/dist/plugins/robots.js.map +0 -1
  90. package/dist/plugins/sitemap.d.ts +0 -11
  91. package/dist/plugins/sitemap.d.ts.map +0 -1
  92. package/dist/plugins/sitemap.js +0 -27
  93. package/dist/plugins/sitemap.js.map +0 -1
  94. package/dist/schemas.d.ts +0 -21
  95. package/dist/schemas.d.ts.map +0 -1
  96. package/dist/schemas.js +0 -25
  97. package/dist/schemas.js.map +0 -1
  98. package/dist/tools/__mocks__/inquirer-prompt.d.ts +0 -20
  99. package/dist/tools/__mocks__/inquirer-prompt.d.ts.map +0 -1
  100. package/dist/tools/__mocks__/inquirer-prompt.js +0 -19
  101. package/dist/tools/__mocks__/inquirer-prompt.js.map +0 -1
  102. package/dist/tools/__mocks__/path.d.ts +0 -5
  103. package/dist/tools/__mocks__/path.d.ts.map +0 -1
  104. package/dist/tools/__mocks__/path.js +0 -7
  105. package/dist/tools/__mocks__/path.js.map +0 -1
  106. package/dist/tools/inquirer-prompt.d.ts +0 -11
  107. package/dist/tools/inquirer-prompt.d.ts.map +0 -1
  108. package/dist/tools/inquirer-prompt.js +0 -139
  109. package/dist/tools/inquirer-prompt.js.map +0 -1
  110. package/dist/tools/path.d.ts +0 -4
  111. package/dist/tools/path.d.ts.map +0 -1
  112. package/dist/tools/path.js +0 -6
  113. package/dist/tools/path.js.map +0 -1
  114. package/dist/tools/tools.d.ts +0 -3
  115. package/dist/tools/tools.d.ts.map +0 -1
  116. package/dist/tools/tools.js +0 -10
  117. package/dist/tools/tools.js.map +0 -1
  118. package/dist/types/config.d.ts +0 -8
  119. package/dist/types/config.d.ts.map +0 -1
  120. package/dist/types/config.js +0 -2
  121. package/dist/types/config.js.map +0 -1
  122. package/init/_data/site.json +0 -5
  123. package/init/_includes/assets/css/styles/search/algolia.css +0 -125
  124. package/init/_includes/assets/css/styles/search/pagefind.css +0 -60
  125. package/init/_includes/assets/css/styles/variables.css +0 -30
  126. package/init/_includes/assets/images/docula-logo-horizontal.webp +0 -0
  127. package/init/_includes/assets/images/logo.svg +0 -10
  128. package/init/_includes/doc.njk +0 -22
  129. package/init/_includes/landing/landingHero.njk +0 -8
  130. package/init/_includes/landing/singlePage.njk +0 -24
  131. package/init/_includes/multipage/header.njk +0 -23
  132. package/init/_includes/multipage/homeHeader.njk +0 -20
  133. package/init/_includes/multipage/main.njk +0 -47
  134. package/init/_includes/multipage/scripts.njk +0 -47
  135. package/init/_includes/multipage/searchEngine.njk +0 -5
  136. package/init/_includes/multipage/searchModal.njk +0 -19
  137. package/init/_includes/multipage/sidebar.njk +0 -50
  138. package/init/_includes/search/algolia.njk +0 -98
  139. package/init/_includes/search/pagefind.njk +0 -66
  140. package/init/_includes/shared/footer.njk +0 -4
  141. package/init/_includes/shared/head.njk +0 -15
  142. package/init/_includes/shared/highlightScript.njk +0 -6
  143. package/init/_includes/shared/homeData.njk +0 -49
  144. package/init/api/Readme.md +0 -3
  145. package/init/docs/Readme.md +0 -83
  146. package/init/docs/docs.json +0 -13
  147. package/init/index.njk +0 -28
  148. package/init/releases.njk +0 -38
  149. package/init/search-index.md +0 -14
  150. package/init/versions.njk +0 -40
  151. /package/{init/_includes/assets → template}/css/highlight/highlight.min.js +0 -0
  152. /package/{init/_includes/assets → template}/css/highlight/styles/base16/dracula.min.css +0 -0
package/README.md CHANGED
@@ -1,51 +1,120 @@
1
- <img src="branding-guidelines/Docula Logos/PNG/Color/docula.png" />
1
+ ![Docula](site/logo.svg)
2
2
 
3
- # Docula
3
+ ---
4
4
 
5
+ ## Beautiful Website for Your Projects
5
6
  [![tests](https://github.com/jaredwray/docula/actions/workflows/tests.yaml/badge.svg)](https://github.com/jaredwray/docula/actions/workflows/tests.yaml)
6
- [![codecov](https://codecov.io/github/jaredwray/docula/branch/main/graph/badge.svg?token=RS0GPY4V4M)](https://codecov.io/github/jaredwray/docula)
7
- [![release](https://github.com/jaredwray/docula/actions/workflows/release.yaml/badge.svg)](https://github.com/jaredwray/docula/actions/workflows/release.yaml)
8
- [![npm](https://img.shields.io/npm/dm/docula.svg)](https://www.npmjs.com/package/docula)
9
- [![npm](https://img.shields.io/npm/v/docula.svg)](https://www.npmjs.com/package/docula)
7
+ [![GitHub license](https://img.shields.io/github/license/jaredwray/docula)](https://github.com/jaredwray/docula/blob/master/LICENSE)
8
+ [![codecov](https://codecov.io/gh/jaredwray/docula/graph/badge.svg?token=RS0GPY4V4M)](https://codecov.io/gh/jaredwray/docula)
9
+ [![npm](https://img.shields.io/npm/dm/docula)](https://npmjs.com/package/docula)
10
+ [![npm](https://img.shields.io/npm/v/docula)](https://npmjs.com/package/docula)
11
+
12
+ ---
13
+ ## Table of Contents
14
+ - [Features](#features)
15
+ - [Getting Started](#getting-started)
16
+ - [Using Your own Template](#using-your-own-template)
17
+ - [Building Multiple Pages](#building-multiple-pages)
18
+ - [Helper Functions for Markdown](#helper-functions-for-markdown)
19
+ - [Code of Conduct and Contributing](#code-of-conduct-and-contributing)
20
+ - [License - MIT](#license)
10
21
 
11
22
  ## Features
23
+ * No configuration requrired. Just setup the folder structure with a logo, favicon, and css file.
24
+ * Builds a static website that can be hosted anywhere.
25
+ * For more complex projects easily add a `docula.config.js` or `docula.config.ts` file to customize the build process. With PRE and POST methods.
26
+ * Support for single page with readme or multiple markdown pages in a docs folder.
27
+ * Will generate a sitemap.xml and robots.txt for your site.
28
+ * Uses Github release notes to generate a changelog / releases page.
29
+ * Uses Github to show contributors and link to their profiles.
30
+ * Simple search is provided by default out of the box.
31
+
32
+ ## Getting Started
33
+
34
+ ## 1. Install docula
35
+
36
+ > npx docula init
12
37
 
13
- - Zero Config - no config needed
14
- - Can run from `npx` or installed globally
15
- - Knowledge Base that is easy to setup and manage
16
- - Beautiful and Responsive Design
17
- - Search Engine Optimized and Plugin (Algolia) and local search by default.
18
- - Convention over Configuration - just add your files, use a template (or build your own) and you're done.
19
- - Will generate a sitemap.xml
20
- - Will generate a robots.txt
21
- - Will generate a release based on Github Releases
22
- - Generatses an RSS feed.xml file to use and share
23
-
24
- ## Site Structure
25
-
26
- - `site/docs` - contains the documentation files
27
- - `site/template` - contains the templates files
28
-
29
- This will all be generated in the `dist` folder by default. To change that you can use the `--output=<folder_name>` flag.
30
-
31
- ### Github Integration
32
- - Pull all of your Contributors and Display Them
33
- - Pulls your release notes and puts it into a good looking data format for your templates
34
- - Pulls your project stats based on Github Forks, Stars, and Watchers
35
- ## How the CLI should work
36
- To create a new project it should be as simple as:
37
- ```bash
38
- docula init <project-name>
38
+ This will create a folder called site with the following structure:
39
+
40
+ ```
41
+ site
42
+ ├───site.css
43
+ ├───logo.png
44
+ ├───favicon.ico
45
+ ├───README.md
46
+ ├───docula.config.js
39
47
  ```
48
+ Note: for typescript do 'docula init --typescript'
49
+
50
+ ## 2. Add your content
51
+
52
+ Simply replace the logo, favicon, and css file with your own. The readme is your root project readme and you just need to at build time move it over to the site folder. If you have it at the root of the project and this is a folder inside just delete the README.md file in the site folder and docula will copy it over for you automatically.
53
+
54
+ ## 3. Build your site
55
+
56
+ > npx docula
57
+
58
+ This will build your site and place it in the `dist` folder. You can then host it anywhere you like.
59
+
60
+ ## Using Your own Template
61
+
62
+ If you want to use your own template you can do so by adding a `docula.config.ts` file to the root of your project. This file will be used to configure the build process.
63
+
64
+ or at the command line:
65
+
66
+ > npx docula --template path/to/template
40
67
 
41
- This will actually do all the scaffolding for the site setting up the package.json, the config file, and the templates. It will also install the default template.
68
+ ## Building Multiple Pages
42
69
 
43
- To build the site just do:
44
- ```bash
45
- docula <path optional>
70
+ If you want to build multiple pages you can easily do that by adding in a `docs` folder to the root of the site folder. Inside of that folder you can add as many pages as you like. Each page will be a markdown file and it will generate a table of contents for you. Here is an example of what it looks like:
71
+
72
+ ```
73
+ site
74
+ ├───site.css
75
+ ├───logo.png
76
+ ├───favicon.ico
77
+ ├───docula.config.ts
78
+ ├───docs
79
+ │ ├───getting-started.md
80
+ │ ├───contributing.md
81
+ │ ├───license.md
82
+ │ ├───code-of-conduct.md
46
83
  ```
47
84
 
48
- To build the site and watch for changes:
49
- ```bash
50
- docula <path optional> --serve --watch --port=8081
85
+ The `readme.md` file will be the root page and the rest will be added to the table of contents. If you want to control the title or order of the pages you can do so by setting the `title` and `order` properties in the front matter of the markdown file. Here is an example:
86
+
87
+ ```md
88
+ ---
89
+ title: Getting Started
90
+ order: 2
91
+ ---
92
+ ```
93
+
94
+ ## Helper Functions for Markdown
95
+
96
+ docula comes with some helper functions that you can use in your markdown files.
97
+ * `doculaHelpers.getFrontMatter(fileName)` - Gets the front matter of a markdown file.
98
+ * `doculaHelpers.setFrontMatter(fileName, frontMatter)` - Sets the front matter of a markdown file.
99
+ * `doculaHelpers.createDoc(source, destination, frontMatter?, contentFn[]?)` - Creates a markdown file with the specified front matter and content. The contentFn is a function that is executed on the original content of the file. This is useful if you want to remove content from the original file.
100
+
101
+ ### Remove html content
102
+
103
+ In some cases your markdown file will have html content in it such as the logo of your project or a badge. You can use the `wrtirHelpers.removeHtmlContent()` helper function to remove that content from the page. Here is an example:
104
+
105
+ ### Get and Set the Front Matter of a Markdown File
106
+
107
+ You can use the `doculaHelpers.getFrontMatter()` and `doculaHelpers.setFrontMatter()` helper functions to get and set the front matter of a markdown file. Here is an example:
108
+
109
+ ```js
110
+ const frontMatter = doculaHelpers.getFrontMatter('../readme.md');
111
+ frontMatter.title = 'My Title';
112
+ doculaHelpers.setFrontMatter('../readme.md', frontMatter);
51
113
  ```
114
+
115
+ ## Code of Conduct and Contributing
116
+ [Code of Conduct](CODE_OF_CONDUCT.md) and [Contributing](CONTRIBUTING.md) guidelines.
117
+
118
+ ## License
119
+
120
+ MIT © [Jared Wray](https://jaredwray.com)
package/bin/docula.mjs ADDED
@@ -0,0 +1,8 @@
1
+ #!/usr/bin/env ts-node
2
+
3
+ import * as process from 'node:process';
4
+ import Docula from '../dist/docula.js';
5
+
6
+ const docula = new Docula();
7
+
8
+ await docula.execute(process);
@@ -0,0 +1,34 @@
1
+ import { DoculaOptions } from './options.js';
2
+ import { type GithubData } from './github.js';
3
+ export type DoculaData = {
4
+ siteUrl: string;
5
+ siteTitle: string;
6
+ siteDescription: string;
7
+ sitePath: string;
8
+ templatePath: string;
9
+ outputPath: string;
10
+ githubPath?: string;
11
+ github?: GithubData;
12
+ templates?: DoculaTemplates;
13
+ };
14
+ export type DoculaTemplates = {
15
+ index: string;
16
+ releases: string;
17
+ };
18
+ export declare class DoculaBuilder {
19
+ private readonly _options;
20
+ private readonly _ecto;
21
+ private readonly _console;
22
+ constructor(options?: DoculaOptions, engineOptions?: any);
23
+ get options(): DoculaOptions;
24
+ build(): Promise<void>;
25
+ validateOptions(options: DoculaOptions): void;
26
+ getGithubData(githubPath: string): Promise<GithubData>;
27
+ getTemplates(options: DoculaOptions): Promise<DoculaTemplates>;
28
+ getTemplateFile(path: string, name: string): Promise<string | undefined>;
29
+ buildRobotsPage(options: DoculaOptions): Promise<void>;
30
+ buildSiteMapPage(data: DoculaData): Promise<void>;
31
+ buildIndexPage(data: DoculaData): Promise<void>;
32
+ buildReleasePage(data: DoculaData): Promise<void>;
33
+ buildReadmeSection(data: DoculaData): Promise<string>;
34
+ }
@@ -0,0 +1,182 @@
1
+ import { Ecto } from 'ecto';
2
+ import fs from 'fs-extra';
3
+ import { DoculaOptions } from './options.js';
4
+ import { DoculaConsole } from './console.js';
5
+ import { Github } from './github.js';
6
+ export class DoculaBuilder {
7
+ _options = new DoculaOptions();
8
+ _ecto;
9
+ _console = new DoculaConsole();
10
+ constructor(options, engineOptions) {
11
+ if (options) {
12
+ this._options = options;
13
+ }
14
+ // eslint-disable-next-line @typescript-eslint/no-unsafe-argument
15
+ this._ecto = new Ecto(engineOptions);
16
+ }
17
+ get options() {
18
+ return this._options;
19
+ }
20
+ async build() {
21
+ const startTime = Date.now();
22
+ // Validate the options
23
+ this.validateOptions(this.options);
24
+ // Set the site options
25
+ const doculaData = {
26
+ siteUrl: this.options.siteUrl,
27
+ siteTitle: this.options.siteTitle,
28
+ siteDescription: this.options.siteDescription,
29
+ sitePath: this.options.sitePath,
30
+ templatePath: this.options.templatePath,
31
+ outputPath: this.options.outputPath,
32
+ githubPath: this.options.githubPath,
33
+ };
34
+ // Get data from github
35
+ const githubData = await this.getGithubData(this.options.githubPath);
36
+ // Get data of the site
37
+ doculaData.github = githubData;
38
+ // Get the templates to use
39
+ doculaData.templates = await this.getTemplates(this.options);
40
+ // Build the home page (index.html)
41
+ await this.buildIndexPage(doculaData);
42
+ // Build the releases page (/releases/index.html)
43
+ await this.buildReleasePage(doculaData);
44
+ // Build the sitemap (/sitemap.xml)
45
+ await this.buildSiteMapPage(doculaData);
46
+ // Build the robots.txt (/robots.txt)
47
+ await this.buildRobotsPage(this.options);
48
+ const siteRelativePath = this.options.sitePath;
49
+ // Copy over favicon
50
+ if (await fs.pathExists(`${siteRelativePath}/favicon.ico`)) {
51
+ await fs.copy(`${siteRelativePath}/favicon.ico`, `${this.options.outputPath}/favicon.ico`);
52
+ }
53
+ // Copy over logo
54
+ if (await fs.pathExists(`${siteRelativePath}/logo.svg`)) {
55
+ await fs.copy(`${siteRelativePath}/logo.svg`, `${this.options.outputPath}/logo.svg`);
56
+ }
57
+ // Copy over css
58
+ if (await fs.pathExists(`${this.options.templatePath}/css`)) {
59
+ await fs.copy(`${this.options.templatePath}/css`, `${this.options.outputPath}/css`);
60
+ }
61
+ // Copy over variables
62
+ if (await fs.pathExists(`${siteRelativePath}/variables.css`)) {
63
+ await fs.copy(`${siteRelativePath}/variables.css`, `${this.options.outputPath}/css/variables.css`);
64
+ }
65
+ const endTime = Date.now();
66
+ const executionTime = endTime - startTime;
67
+ this._console.log(`Build completed in ${executionTime}ms`);
68
+ }
69
+ validateOptions(options) {
70
+ if (options.githubPath.length < 3) {
71
+ throw new Error('No github options provided');
72
+ }
73
+ if (options.siteDescription.length < 3) {
74
+ throw new Error('No site description options provided');
75
+ }
76
+ if (!options.siteTitle) {
77
+ throw new Error('No site title options provided');
78
+ }
79
+ if (!options.siteUrl) {
80
+ throw new Error('No site url options provided');
81
+ }
82
+ }
83
+ async getGithubData(githubPath) {
84
+ const paths = githubPath.split('/');
85
+ const options = {
86
+ author: paths[0],
87
+ repo: paths[1],
88
+ };
89
+ const github = new Github(options);
90
+ return github.getData();
91
+ }
92
+ async getTemplates(options) {
93
+ const templates = {
94
+ index: '',
95
+ releases: '',
96
+ };
97
+ if (await fs.pathExists(options.templatePath)) {
98
+ const index = await this.getTemplateFile(options.templatePath, 'index');
99
+ if (index) {
100
+ templates.index = index;
101
+ }
102
+ const releases = await this.getTemplateFile(options.templatePath, 'releases');
103
+ if (releases) {
104
+ templates.releases = releases;
105
+ }
106
+ }
107
+ else {
108
+ throw new Error(`No template path found at ${options.templatePath}`);
109
+ }
110
+ return templates;
111
+ }
112
+ async getTemplateFile(path, name) {
113
+ let result;
114
+ const files = await fs.readdir(path);
115
+ for (const file of files) {
116
+ const fileName = file.split('.');
117
+ if (fileName[0].toString().toLowerCase() === name.toLowerCase()) {
118
+ result = file.toString();
119
+ break;
120
+ }
121
+ }
122
+ return result;
123
+ }
124
+ async buildRobotsPage(options) {
125
+ const { sitePath } = options;
126
+ const { outputPath } = options;
127
+ const robotsPath = `${outputPath}/robots.txt`;
128
+ await fs.ensureDir(outputPath);
129
+ await ((await fs.pathExists(`${sitePath}/robots.txt`))
130
+ ? fs.copy(`${sitePath}/robots.txt`, robotsPath)
131
+ : fs.writeFile(robotsPath, 'User-agent: *\nDisallow:'));
132
+ }
133
+ async buildSiteMapPage(data) {
134
+ const sitemapPath = `${data.outputPath}/sitemap.xml`;
135
+ const urls = [{ url: data.siteUrl }, { url: `${data.siteUrl}/releases` }];
136
+ let xml = '<?xml version="1.0" encoding="UTF-8"?>';
137
+ xml += '<urlset xmlns="http://www.sitemaps.org/schemas/sitemap/0.9">';
138
+ for (const { url } of urls) {
139
+ xml += '<url>';
140
+ xml += `<loc>${url}</loc>`;
141
+ xml += '</url>';
142
+ }
143
+ xml += '</urlset>';
144
+ await fs.ensureDir(data.outputPath);
145
+ await fs.writeFile(sitemapPath, xml, 'utf8');
146
+ }
147
+ async buildIndexPage(data) {
148
+ if (data.templates) {
149
+ const indexPath = `${data.outputPath}/index.html`;
150
+ await fs.ensureDir(data.outputPath);
151
+ const indexTemplate = `${data.templatePath}/${data.templates.index}`;
152
+ const htmlReadme = await this.buildReadmeSection(data);
153
+ const indexContent = await this._ecto.renderFromFile(indexTemplate, { ...data, content: htmlReadme }, data.templatePath);
154
+ await fs.writeFile(indexPath, indexContent, 'utf8');
155
+ }
156
+ else {
157
+ throw new Error('No templates found');
158
+ }
159
+ }
160
+ async buildReleasePage(data) {
161
+ if (data.github && data.templates) {
162
+ const releasesPath = `${data.outputPath}/releases/index.html`;
163
+ const releaseOutputPath = `${data.outputPath}/releases`;
164
+ await fs.ensureDir(releaseOutputPath);
165
+ const releasesTemplate = `${data.templatePath}/${data.templates.releases}`;
166
+ const releasesContent = await this._ecto.renderFromFile(releasesTemplate, data, data.templatePath);
167
+ await fs.writeFile(releasesPath, releasesContent, 'utf8');
168
+ }
169
+ else {
170
+ throw new Error('No github data found');
171
+ }
172
+ }
173
+ async buildReadmeSection(data) {
174
+ let htmlReadme = '';
175
+ if (fs.existsSync(`${data.sitePath}/README.md`)) {
176
+ const readmeContent = fs.readFileSync(`${data.sitePath}/README.md`, 'utf8');
177
+ htmlReadme = await this._ecto.markdown.render(readmeContent);
178
+ }
179
+ return htmlReadme;
180
+ }
181
+ }
182
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"builder.js","sourceRoot":"","sources":["../src/builder.ts"],"names":[],"mappings":"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","sourcesContent":["import {Ecto} from 'ecto';\nimport fs from 'fs-extra';\nimport {DoculaOptions} from './options.js';\nimport {DoculaConsole} from './console.js';\nimport {Github, type GithubData, type GithubOptions} from './github.js';\n\nexport type DoculaData = {\n\tsiteUrl: string;\n\tsiteTitle: string;\n\tsiteDescription: string;\n\tsitePath: string;\n\ttemplatePath: string;\n\toutputPath: string;\n\tgithubPath?: string;\n\tgithub?: GithubData;\n\ttemplates?: DoculaTemplates;\n};\n\nexport type DoculaTemplates = {\n\tindex: string;\n\treleases: string;\n};\n\nexport class DoculaBuilder {\n\tprivate readonly _options: DoculaOptions = new DoculaOptions();\n\tprivate readonly _ecto: Ecto;\n\tprivate readonly _console: DoculaConsole = new DoculaConsole();\n\n\tconstructor(options?: DoculaOptions, engineOptions?: any) {\n\t\tif (options) {\n\t\t\tthis._options = options;\n\t\t}\n\n\t\t// eslint-disable-next-line @typescript-eslint/no-unsafe-argument\n\t\tthis._ecto = new Ecto(engineOptions);\n\t}\n\n\tpublic get options(): DoculaOptions {\n\t\treturn this._options;\n\t}\n\n\tpublic async build(): Promise<void> {\n\t\tconst startTime = Date.now();\n\n\t\t// Validate the options\n\t\tthis.validateOptions(this.options);\n\t\t// Set the site options\n\t\tconst doculaData: DoculaData = {\n\t\t\tsiteUrl: this.options.siteUrl,\n\t\t\tsiteTitle: this.options.siteTitle,\n\t\t\tsiteDescription: this.options.siteDescription,\n\t\t\tsitePath: this.options.sitePath,\n\t\t\ttemplatePath: this.options.templatePath,\n\t\t\toutputPath: this.options.outputPath,\n\t\t\tgithubPath: this.options.githubPath,\n\t\t};\n\n\t\t// Get data from github\n\t\tconst githubData = await this.getGithubData(this.options.githubPath);\n\t\t// Get data of the site\n\t\tdoculaData.github = githubData;\n\t\t// Get the templates to use\n\t\tdoculaData.templates = await this.getTemplates(this.options);\n\n\t\t// Build the home page (index.html)\n\t\tawait this.buildIndexPage(doculaData);\n\n\t\t// Build the releases page (/releases/index.html)\n\t\tawait this.buildReleasePage(doculaData);\n\n\t\t// Build the sitemap (/sitemap.xml)\n\t\tawait this.buildSiteMapPage(doculaData);\n\n\t\t// Build the robots.txt (/robots.txt)\n\t\tawait this.buildRobotsPage(this.options);\n\n\t\tconst siteRelativePath = this.options.sitePath;\n\n\t\t// Copy over favicon\n\t\tif (await fs.pathExists(`${siteRelativePath}/favicon.ico`)) {\n\t\t\tawait fs.copy(\n\t\t\t\t`${siteRelativePath}/favicon.ico`,\n\t\t\t\t`${this.options.outputPath}/favicon.ico`,\n\t\t\t);\n\t\t}\n\n\t\t// Copy over logo\n\t\tif (await fs.pathExists(`${siteRelativePath}/logo.svg`)) {\n\t\t\tawait fs.copy(\n\t\t\t\t`${siteRelativePath}/logo.svg`,\n\t\t\t\t`${this.options.outputPath}/logo.svg`,\n\t\t\t);\n\t\t}\n\n\t\t// Copy over css\n\t\tif (await fs.pathExists(`${this.options.templatePath}/css`)) {\n\t\t\tawait fs.copy(\n\t\t\t\t`${this.options.templatePath}/css`,\n\t\t\t\t`${this.options.outputPath}/css`,\n\t\t\t);\n\t\t}\n\n\t\t// Copy over variables\n\t\tif (await fs.pathExists(`${siteRelativePath}/variables.css`)) {\n\t\t\tawait fs.copy(\n\t\t\t\t`${siteRelativePath}/variables.css`,\n\t\t\t\t`${this.options.outputPath}/css/variables.css`,\n\t\t\t);\n\t\t}\n\n\t\tconst endTime = Date.now();\n\n\t\tconst executionTime = endTime - startTime;\n\n\t\tthis._console.log(`Build completed in ${executionTime}ms`);\n\t}\n\n\tpublic validateOptions(options: DoculaOptions): void {\n\t\tif (options.githubPath.length < 3) {\n\t\t\tthrow new Error('No github options provided');\n\t\t}\n\n\t\tif (options.siteDescription.length < 3) {\n\t\t\tthrow new Error('No site description options provided');\n\t\t}\n\n\t\tif (!options.siteTitle) {\n\t\t\tthrow new Error('No site title options provided');\n\t\t}\n\n\t\tif (!options.siteUrl) {\n\t\t\tthrow new Error('No site url options provided');\n\t\t}\n\t}\n\n\tpublic async getGithubData(githubPath: string): Promise<GithubData> {\n\t\tconst paths = githubPath.split('/');\n\t\tconst options: GithubOptions = {\n\t\t\tauthor: paths[0],\n\t\t\trepo: paths[1],\n\t\t};\n\t\tconst github = new Github(options);\n\t\treturn github.getData();\n\t}\n\n\tpublic async getTemplates(options: DoculaOptions): Promise<DoculaTemplates> {\n\t\tconst templates: DoculaTemplates = {\n\t\t\tindex: '',\n\t\t\treleases: '',\n\t\t};\n\n\t\tif (await fs.pathExists(options.templatePath)) {\n\t\t\tconst index = await this.getTemplateFile(options.templatePath, 'index');\n\t\t\tif (index) {\n\t\t\t\ttemplates.index = index;\n\t\t\t}\n\n\t\t\tconst releases = await this.getTemplateFile(\n\t\t\t\toptions.templatePath,\n\t\t\t\t'releases',\n\t\t\t);\n\t\t\tif (releases) {\n\t\t\t\ttemplates.releases = releases;\n\t\t\t}\n\t\t} else {\n\t\t\tthrow new Error(`No template path found at ${options.templatePath}`);\n\t\t}\n\n\t\treturn templates;\n\t}\n\n\tpublic async getTemplateFile(\n\t\tpath: string,\n\t\tname: string,\n\t): Promise<string | undefined> {\n\t\tlet result;\n\t\tconst files = await fs.readdir(path);\n\t\tfor (const file of files) {\n\t\t\tconst fileName = file.split('.');\n\t\t\tif (fileName[0].toString().toLowerCase() === name.toLowerCase()) {\n\t\t\t\tresult = file.toString();\n\t\t\t\tbreak;\n\t\t\t}\n\t\t}\n\n\t\treturn result;\n\t}\n\n\tpublic async buildRobotsPage(options: DoculaOptions): Promise<void> {\n\t\tconst {sitePath} = options;\n\t\tconst {outputPath} = options;\n\t\tconst robotsPath = `${outputPath}/robots.txt`;\n\n\t\tawait fs.ensureDir(outputPath);\n\n\t\tawait ((await fs.pathExists(`${sitePath}/robots.txt`))\n\t\t\t? fs.copy(`${sitePath}/robots.txt`, robotsPath)\n\t\t\t: fs.writeFile(robotsPath, 'User-agent: *\\nDisallow:'));\n\t}\n\n\tpublic async buildSiteMapPage(data: DoculaData): Promise<void> {\n\t\tconst sitemapPath = `${data.outputPath}/sitemap.xml`;\n\t\tconst urls = [{url: data.siteUrl}, {url: `${data.siteUrl}/releases`}];\n\n\t\tlet xml = '<?xml version=\"1.0\" encoding=\"UTF-8\"?>';\n\t\txml += '<urlset xmlns=\"http://www.sitemaps.org/schemas/sitemap/0.9\">';\n\n\t\tfor (const {url} of urls) {\n\t\t\txml += '<url>';\n\t\t\txml += `<loc>${url}</loc>`;\n\t\t\txml += '</url>';\n\t\t}\n\n\t\txml += '</urlset>';\n\n\t\tawait fs.ensureDir(data.outputPath);\n\n\t\tawait fs.writeFile(sitemapPath, xml, 'utf8');\n\t}\n\n\tpublic async buildIndexPage(data: DoculaData): Promise<void> {\n\t\tif (data.templates) {\n\t\t\tconst indexPath = `${data.outputPath}/index.html`;\n\n\t\t\tawait fs.ensureDir(data.outputPath);\n\n\t\t\tconst indexTemplate = `${data.templatePath}/${data.templates.index}`;\n\n\t\t\tconst htmlReadme = await this.buildReadmeSection(data);\n\n\t\t\tconst indexContent = await this._ecto.renderFromFile(\n\t\t\t\tindexTemplate,\n\t\t\t\t{...data, content: htmlReadme},\n\t\t\t\tdata.templatePath,\n\t\t\t);\n\t\t\tawait fs.writeFile(indexPath, indexContent, 'utf8');\n\t\t} else {\n\t\t\tthrow new Error('No templates found');\n\t\t}\n\t}\n\n\tpublic async buildReleasePage(data: DoculaData): Promise<void> {\n\t\tif (data.github && data.templates) {\n\t\t\tconst releasesPath = `${data.outputPath}/releases/index.html`;\n\t\t\tconst releaseOutputPath = `${data.outputPath}/releases`;\n\n\t\t\tawait fs.ensureDir(releaseOutputPath);\n\n\t\t\tconst releasesTemplate = `${data.templatePath}/${data.templates.releases}`;\n\t\t\tconst releasesContent = await this._ecto.renderFromFile(\n\t\t\t\treleasesTemplate,\n\t\t\t\tdata,\n\t\t\t\tdata.templatePath,\n\t\t\t);\n\t\t\tawait fs.writeFile(releasesPath, releasesContent, 'utf8');\n\t\t} else {\n\t\t\tthrow new Error('No github data found');\n\t\t}\n\t}\n\n\tpublic async buildReadmeSection(data: DoculaData): Promise<string> {\n\t\tlet htmlReadme = '';\n\t\tif (fs.existsSync(`${data.sitePath}/README.md`)) {\n\t\t\tconst readmeContent = fs.readFileSync(\n\t\t\t\t`${data.sitePath}/README.md`,\n\t\t\t\t'utf8',\n\t\t\t);\n\t\t\thtmlReadme = await this._ecto.markdown.render(readmeContent);\n\t\t}\n\n\t\treturn htmlReadme;\n\t}\n}\n"]}
@@ -0,0 +1,22 @@
1
+ export declare class DoculaConsole {
2
+ log(message: string): void;
3
+ error(message: string): void;
4
+ warn(message: string): void;
5
+ printHelp(): void;
6
+ parseProcessArgv(argv: string[]): DoculaConsoleProcess;
7
+ getCommand(argv: string[]): string | undefined;
8
+ getArguments(argv: string[]): DoculaConsoleArguments;
9
+ }
10
+ type DoculaConsoleProcess = {
11
+ argv: string[];
12
+ command: string | undefined;
13
+ args: DoculaConsoleArguments;
14
+ };
15
+ type DoculaConsoleArguments = {
16
+ sitePath: string | undefined;
17
+ templatePath: string | undefined;
18
+ output: string | undefined;
19
+ watch: boolean;
20
+ port: number;
21
+ };
22
+ export {};
@@ -0,0 +1,120 @@
1
+ import path from 'node:path';
2
+ import process from 'node:process';
3
+ export class DoculaConsole {
4
+ log(message) {
5
+ console.log(message);
6
+ }
7
+ error(message) {
8
+ console.error(message);
9
+ }
10
+ warn(message) {
11
+ console.warn(message);
12
+ }
13
+ printHelp() {
14
+ console.log(' Usage: docula [command] [arguments]');
15
+ console.log();
16
+ console.log(' Commands:');
17
+ console.log(' init Initialize a new project');
18
+ console.log(' build Build the project. By default just npx docula will build the project if it finds a ./site folder');
19
+ console.log(' serve Serve the project as a local website');
20
+ console.log(' help Print this help');
21
+ console.log(' version Print the version');
22
+ console.log();
23
+ console.log(' Arguments Build:');
24
+ console.log(' -w, --watch watch for changes and rebuild');
25
+ console.log(' -s, --site Set the path where site files are located');
26
+ console.log(' -o, --outputPath Set the output directory. Default is ./site/dist');
27
+ console.log(' -t, --templatePath Set the custom template to use');
28
+ console.log();
29
+ console.log(' Arguments serve:');
30
+ console.log(' -p, --port Set the port number used with serve');
31
+ console.log(' -w, --watch watch for changes and rebuild');
32
+ console.log(' -s, --site Set the path where site files are located');
33
+ }
34
+ parseProcessArgv(argv) {
35
+ const command = this.getCommand(argv);
36
+ const args = this.getArguments(argv);
37
+ return {
38
+ argv,
39
+ command,
40
+ args,
41
+ };
42
+ }
43
+ getCommand(argv) {
44
+ let result;
45
+ for (const arg of argv) {
46
+ // eslint-disable-next-line default-case
47
+ switch (arg) {
48
+ case 'init': {
49
+ result = 'init';
50
+ break;
51
+ }
52
+ case 'build': {
53
+ result = 'build';
54
+ break;
55
+ }
56
+ case 'serve': {
57
+ result = 'serve';
58
+ break;
59
+ }
60
+ case 'help': {
61
+ result = 'help';
62
+ break;
63
+ }
64
+ case 'version': {
65
+ result = arg;
66
+ break;
67
+ }
68
+ }
69
+ }
70
+ return result;
71
+ }
72
+ getArguments(argv) {
73
+ const args = {
74
+ sitePath: '',
75
+ templatePath: '',
76
+ output: '',
77
+ watch: false,
78
+ port: 3000,
79
+ };
80
+ for (let i = 0; i < argv.length; i++) {
81
+ const arg = argv[i];
82
+ // eslint-disable-next-line default-case
83
+ switch (arg) {
84
+ case '-p':
85
+ case '--port': {
86
+ const portString = argv[i + 1];
87
+ if (portString !== undefined) {
88
+ args.port = Number.parseInt(portString, 10);
89
+ }
90
+ break;
91
+ }
92
+ case '-o':
93
+ case '--output': {
94
+ args.output = argv[i + 1];
95
+ args.output = path.join(process.cwd(), args.output);
96
+ break;
97
+ }
98
+ case '-w':
99
+ case '--watch': {
100
+ args.watch = true;
101
+ break;
102
+ }
103
+ case '-s':
104
+ case '--site': {
105
+ args.sitePath = argv[i + 1];
106
+ args.sitePath = path.join(process.cwd(), args.sitePath);
107
+ break;
108
+ }
109
+ case '-t':
110
+ case '--templatePath': {
111
+ args.templatePath = argv[i + 1];
112
+ args.templatePath = path.join(process.cwd(), args.templatePath);
113
+ break;
114
+ }
115
+ }
116
+ }
117
+ return args;
118
+ }
119
+ }
120
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"console.js","sourceRoot":"","sources":["../src/console.ts"],"names":[],"mappings":"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","sourcesContent":["import path from 'node:path';\nimport process from 'node:process';\n\nexport class DoculaConsole {\n\tlog(message: string): void {\n\t\tconsole.log(message);\n\t}\n\n\terror(message: string): void {\n\t\tconsole.error(message);\n\t}\n\n\twarn(message: string): void {\n\t\tconsole.warn(message);\n\t}\n\n\tprintHelp(): void {\n\t\tconsole.log('   Usage: docula [command] [arguments]');\n\t\tconsole.log();\n\t\tconsole.log('   Commands:');\n\t\tconsole.log('     init           Initialize a new project');\n\t\tconsole.log('     build          Build the project. By default just npx docula will build the project if it finds a ./site folder');\n\t\tconsole.log('     serve          Serve the project as a local website');\n\t\tconsole.log('     help           Print this help');\n\t\tconsole.log('     version        Print the version');\n\t\tconsole.log();\n\t\tconsole.log('   Arguments Build:');\n\t\tconsole.log('     -w, --watch            watch for changes and rebuild');\n\t\tconsole.log('     -s, --site             Set the path where site files are located');\n\t\tconsole.log('     -o, --outputPath       Set the output directory. Default is ./site/dist');\n\t\tconsole.log('     -t, --templatePath     Set the custom template to use');\n\t\tconsole.log();\n\t\tconsole.log('   Arguments serve:');\n\t\tconsole.log('     -p, --port         Set the port number used with serve');\n\t\tconsole.log('     -w, --watch        watch for changes and rebuild');\n\t\tconsole.log('     -s, --site         Set the path where site files are located');\n\t}\n\n\tpublic parseProcessArgv(argv: string[]): DoculaConsoleProcess {\n\t\tconst command = this.getCommand(argv);\n\t\tconst args = this.getArguments(argv);\n\t\treturn {\n\t\t\targv,\n\t\t\tcommand,\n\t\t\targs,\n\t\t};\n\t}\n\n\tpublic getCommand(argv: string[]): string | undefined {\n\t\tlet result;\n\t\tfor (const arg of argv) {\n\t\t\t// eslint-disable-next-line default-case\n\t\t\tswitch (arg) {\n\t\t\t\tcase 'init': {\n\t\t\t\t\tresult = 'init';\n\t\t\t\t\tbreak;\n\t\t\t\t}\n\n\t\t\t\tcase 'build': {\n\t\t\t\t\tresult = 'build';\n\t\t\t\t\tbreak;\n\t\t\t\t}\n\n\t\t\t\tcase 'serve': {\n\t\t\t\t\tresult = 'serve';\n\t\t\t\t\tbreak;\n\t\t\t\t}\n\n\t\t\t\tcase 'help': {\n\t\t\t\t\tresult = 'help';\n\t\t\t\t\tbreak;\n\t\t\t\t}\n\n\t\t\t\tcase 'version': {\n\t\t\t\t\tresult = arg;\n\t\t\t\t\tbreak;\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\n\t\treturn result;\n\t}\n\n\tpublic getArguments(argv: string[]): DoculaConsoleArguments {\n\t\tconst args = {\n\t\t\tsitePath: '',\n\t\t\ttemplatePath: '',\n\t\t\toutput: '',\n\t\t\twatch: false,\n\t\t\tport: 3000,\n\t\t};\n\t\tfor (let i = 0; i < argv.length; i++) {\n\t\t\tconst arg = argv[i];\n\t\t\t// eslint-disable-next-line default-case\n\t\t\tswitch (arg) {\n\t\t\t\tcase '-p':\n\t\t\t\tcase '--port': {\n\t\t\t\t\tconst portString = argv[i + 1];\n\t\t\t\t\tif (portString !== undefined) {\n\t\t\t\t\t\targs.port = Number.parseInt(portString, 10);\n\t\t\t\t\t}\n\n\t\t\t\t\tbreak;\n\t\t\t\t}\n\n\t\t\t\tcase '-o':\n\t\t\t\tcase '--output': {\n\t\t\t\t\targs.output = argv[i + 1];\n\t\t\t\t\targs.output = path.join(process.cwd(), args.output);\n\t\t\t\t\tbreak;\n\t\t\t\t}\n\n\t\t\t\tcase '-w':\n\t\t\t\tcase '--watch': {\n\t\t\t\t\targs.watch = true;\n\t\t\t\t\tbreak;\n\t\t\t\t}\n\n\t\t\t\tcase '-s':\n\t\t\t\tcase '--site': {\n\t\t\t\t\targs.sitePath = argv[i + 1];\n\t\t\t\t\targs.sitePath = path.join(process.cwd(), args.sitePath);\n\t\t\t\t\tbreak;\n\t\t\t\t}\n\n\t\t\t\tcase '-t':\n\t\t\t\tcase '--templatePath': {\n\t\t\t\t\targs.templatePath = argv[i + 1];\n\t\t\t\t\targs.templatePath = path.join(process.cwd(), args.templatePath);\n\t\t\t\t\tbreak;\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\n\t\treturn args;\n\t}\n}\n\ntype DoculaConsoleProcess = {\n\targv: string[];\n\tcommand: string | undefined;\n\targs: DoculaConsoleArguments;\n};\n\ntype DoculaConsoleArguments = {\n\tsitePath: string | undefined;\n\ttemplatePath: string | undefined;\n\toutput: string | undefined;\n\twatch: boolean;\n\tport: number;\n};\n\n"]}
package/dist/docula.d.ts CHANGED
@@ -1,22 +1,23 @@
1
- import { Config } from './config.js';
2
- import type { CommanderOptions } from './index.js';
3
- export declare class Docula {
4
- readonly config: Config;
5
- private readonly eleventy;
6
- private pluginInstances;
7
- private readonly beforePlugins;
8
- private readonly afterPlugins;
9
- private readonly landingFilesExceptions;
10
- constructor(options?: CommanderOptions);
11
- init(sitePath?: string): Promise<void>;
12
- build(): Promise<void>;
13
- serve(): Promise<void>;
14
- copyLandingFolder(source: string, target: string): void;
15
- copyFolder(source: string, target: string): void;
16
- copySearchEngineFiles(): void;
17
- writeConfigFile(sitePath?: string): Promise<Record<string, unknown>>;
18
- validateFilePath(source: string, target: string): Record<string, any>;
19
- private loadPlugins;
20
- private readonly executePlugins;
1
+ /// <reference types="node" />
2
+ /// <reference types="node" />
3
+ import type http from 'node:http';
4
+ import { DoculaOptions } from './options.js';
5
+ export default class Docula {
6
+ private _options;
7
+ private readonly _console;
8
+ private _configFileModule;
9
+ private _server;
10
+ constructor(options?: DoculaOptions);
11
+ get options(): DoculaOptions;
12
+ set options(value: DoculaOptions);
13
+ get server(): http.Server | undefined;
14
+ get configFileModule(): any;
15
+ checkForUpdates(): void;
16
+ execute(process: NodeJS.Process): Promise<void>;
17
+ isSinglePageWebsite(sitePath: string): boolean;
18
+ generateInit(sitePath: string): void;
19
+ getVersion(): string;
20
+ loadConfigFile(sitePath: string): Promise<void>;
21
+ serve(options: DoculaOptions): Promise<void>;
21
22
  }
22
- //# sourceMappingURL=docula.d.ts.map
23
+ export { DoculaHelpers } from './helpers.js';