writr 1.9.12 → 2.0.1

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 (133) hide show
  1. package/README.md +76 -135
  2. package/bin/writr.mjs +8 -0
  3. package/dist/builder.d.ts +33 -0
  4. package/dist/builder.js +181 -0
  5. package/dist/console.d.ts +22 -0
  6. package/dist/console.js +120 -0
  7. package/dist/github.d.ts +22 -0
  8. package/dist/github.js +74 -0
  9. package/dist/helpers/markdown.d.ts +10 -0
  10. package/dist/helpers/markdown.js +24 -0
  11. package/dist/helpers.d.ts +7 -0
  12. package/dist/helpers.js +58 -0
  13. package/dist/options.d.ts +12 -0
  14. package/dist/options.js +47 -0
  15. package/dist/writr.d.ts +23 -0
  16. package/dist/writr.js +153 -0
  17. package/init/favicon.svg +1 -0
  18. package/init/logo.png +0 -0
  19. package/init/variables.css +30 -0
  20. package/init/writr.config.cjs +10 -0
  21. package/package.json +44 -66
  22. package/template/css/highlight/highlight.min.js +1433 -0
  23. package/template/css/highlight/styles/base16/dracula.min.css +7 -0
  24. package/template/css/home.css +307 -0
  25. package/template/css/index.css +919 -0
  26. package/template/includes/footer.hbs +10 -0
  27. package/template/includes/header.hbs +17 -0
  28. package/template/includes/landing/content.hbs +63 -0
  29. package/template/includes/landing/hero.hbs +3 -0
  30. package/template/includes/scripts.hbs +4 -0
  31. package/template/index.hbs +16 -0
  32. package/template/releases.hbs +45 -0
  33. package/bin/writr.js +0 -7
  34. package/dist/cache.d.ts +0 -18
  35. package/dist/cache.js +0 -70
  36. package/dist/cache.js.map +0 -1
  37. package/dist/config.d.ts +0 -29
  38. package/dist/config.js +0 -104
  39. package/dist/config.js.map +0 -1
  40. package/dist/data/dataProviderInterface.d.ts +0 -12
  41. package/dist/data/dataProviderInterface.js +0 -2
  42. package/dist/data/dataProviderInterface.js.map +0 -1
  43. package/dist/data/dataService.d.ts +0 -19
  44. package/dist/data/dataService.js +0 -130
  45. package/dist/data/dataService.js.map +0 -1
  46. package/dist/data/fileDataProvider.d.ts +0 -20
  47. package/dist/data/fileDataProvider.js +0 -132
  48. package/dist/data/fileDataProvider.js.map +0 -1
  49. package/dist/generator.d.ts +0 -8
  50. package/dist/generator.js +0 -54
  51. package/dist/generator.js.map +0 -1
  52. package/dist/index.d.ts +0 -8
  53. package/dist/index.js +0 -80
  54. package/dist/index.js.map +0 -1
  55. package/dist/log.d.ts +0 -6
  56. package/dist/log.js +0 -13
  57. package/dist/log.js.map +0 -1
  58. package/dist/migrate/ghostMigrationProvider.d.ts +0 -9
  59. package/dist/migrate/ghostMigrationProvider.js +0 -65
  60. package/dist/migrate/ghostMigrationProvider.js.map +0 -1
  61. package/dist/migrate/jekyllMigrationProvider.d.ts +0 -4
  62. package/dist/migrate/jekyllMigrationProvider.js +0 -11
  63. package/dist/migrate/jekyllMigrationProvider.js.map +0 -1
  64. package/dist/migrate/mediumMigrationProvider.d.ts +0 -7
  65. package/dist/migrate/mediumMigrationProvider.js +0 -26
  66. package/dist/migrate/mediumMigrationProvider.js.map +0 -1
  67. package/dist/migrate/migrationProviderInterface.d.ts +0 -3
  68. package/dist/migrate/migrationProviderInterface.js +0 -2
  69. package/dist/migrate/migrationProviderInterface.js.map +0 -1
  70. package/dist/migrate/wordpressMigrationProvider.d.ts +0 -12
  71. package/dist/migrate/wordpressMigrationProvider.js +0 -103
  72. package/dist/migrate/wordpressMigrationProvider.js.map +0 -1
  73. package/dist/migrate.d.ts +0 -6
  74. package/dist/migrate.js +0 -34
  75. package/dist/migrate.js.map +0 -1
  76. package/dist/post.d.ts +0 -31
  77. package/dist/post.js +0 -189
  78. package/dist/post.js.map +0 -1
  79. package/dist/render/atomRenderProvider.d.ts +0 -8
  80. package/dist/render/atomRenderProvider.js +0 -51
  81. package/dist/render/atomRenderProvider.js.map +0 -1
  82. package/dist/render/htmRenderlProvider.d.ts +0 -14
  83. package/dist/render/htmRenderlProvider.js +0 -117
  84. package/dist/render/htmRenderlProvider.js.map +0 -1
  85. package/dist/render/imageRenderProvider.d.ts +0 -8
  86. package/dist/render/imageRenderProvider.js +0 -15
  87. package/dist/render/imageRenderProvider.js.map +0 -1
  88. package/dist/render/jsonRenderProvider.d.ts +0 -8
  89. package/dist/render/jsonRenderProvider.js +0 -24
  90. package/dist/render/jsonRenderProvider.js.map +0 -1
  91. package/dist/render/renderProviderInterface.d.ts +0 -5
  92. package/dist/render/renderProviderInterface.js +0 -2
  93. package/dist/render/renderProviderInterface.js.map +0 -1
  94. package/dist/serve.d.ts +0 -16
  95. package/dist/serve.js +0 -29
  96. package/dist/serve.js.map +0 -1
  97. package/dist/storage/fileStorageProvider.d.ts +0 -10
  98. package/dist/storage/fileStorageProvider.js +0 -62
  99. package/dist/storage/fileStorageProvider.js.map +0 -1
  100. package/dist/storage/storageProviderInterface.d.ts +0 -8
  101. package/dist/storage/storageProviderInterface.js +0 -2
  102. package/dist/storage/storageProviderInterface.js.map +0 -1
  103. package/dist/storage/storageService.d.ts +0 -12
  104. package/dist/storage/storageService.js +0 -28
  105. package/dist/storage/storageService.js.map +0 -1
  106. package/dist/tag.d.ts +0 -9
  107. package/dist/tag.js +0 -34
  108. package/dist/tag.js.map +0 -1
  109. package/dist/utils/parser.d.ts +0 -5
  110. package/dist/utils/parser.js +0 -37
  111. package/dist/utils/parser.js.map +0 -1
  112. package/dist/utils/setup.d.ts +0 -8
  113. package/dist/utils/setup.js +0 -75
  114. package/dist/utils/setup.js.map +0 -1
  115. package/init/article-complex.md +0 -40
  116. package/init/article-ejs.md +0 -40
  117. package/init/article-unpublished.md +0 -11
  118. package/init/article1-simple.md +0 -13
  119. package/init/article1.md +0 -11
  120. package/init/article2.md +0 -14
  121. package/init/big-content.md +0 -11
  122. package/init/config.json +0 -9
  123. package/init/images/Introducing Docula 1.png +0 -0
  124. package/init/images/Introducing Docula 2.jpeg +0 -0
  125. package/init/images/Introducing Docula 3.png +0 -0
  126. package/init/images/Introducing Docula 4.png +0 -0
  127. package/init/permalink-test.md +0 -13
  128. package/init/templates/index.hjs +0 -13
  129. package/init/templates/partials/header.hjs +0 -2
  130. package/init/templates/post.hjs +0 -19
  131. package/init/templates/post2.hjs +0 -17
  132. package/init/templates/post3.ejs +0 -7
  133. package/init/templates/tag.hjs +0 -9
package/README.md CHANGED
@@ -1,183 +1,124 @@
1
- ![Writr](logo.png)
1
+ ![Writr](site/logo.svg)
2
2
 
3
3
  ---
4
4
 
5
- ### CLI Tool for Markdown
5
+ ## Beautiful Website for Your Projects
6
6
  [![Build](https://github.com/jaredwray/writr/actions/workflows/tests.yml/badge.svg)](https://github.com/jaredwray/writr/actions/workflows/tests.yml)
7
7
  [![GitHub license](https://img.shields.io/github/license/jaredwray/writr)](https://github.com/jaredwray/writr/blob/master/LICENSE)
8
8
  [![codecov](https://codecov.io/gh/jaredwray/writr/branch/master/graph/badge.svg?token=1YdMesM07X)](https://codecov.io/gh/jaredwray/writr)
9
9
  [![npm](https://img.shields.io/npm/dm/writr)](https://npmjs.com/package/writr)
10
10
 
11
11
  ---
12
+ ## Table of Contents
13
+ - [Features](#features)
14
+ - [Getting Started](#getting-started)
15
+ - [Using Your own Template](#using-your-own-template)
16
+ - [Building Multiple Pages](#building-multiple-pages)
17
+ - [Helper Functions for Markdown](#helper-functions-for-markdown)
18
+ - [Code of Conduct and Contributing](#code-of-conduct-and-contributing)
19
+ - [What Happened to it Generating a Blog](#what-happened-to-it-generating-a-blog)
20
+ - [License - MIT](#license)
21
+
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 `writr.config.js` or `writr.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.
12
31
 
13
32
  ## Getting Started
14
33
 
15
34
  ## 1. Install Writr
16
35
 
17
- > npm install -g writr
36
+ > npx writr init
18
37
 
19
- ## 2. Setup your directory (look at /blog_example for how to do this)
20
-
21
- > writr init
38
+ This will create a folder called site with the following structure:
22
39
 
23
40
  ```
24
- blog/*.md //markdown files in the folder root
25
- blog/images //images for the blog
26
- blog/config.json //config file (optional)
27
- blog/templates //template directory for your index, post, and tag
41
+ site
42
+ ├───site.css
43
+ ├───logo.png
44
+ ├───favicon.ico
45
+ ├───README.md
46
+ ├───writr.config.js
28
47
  ```
48
+ Note: for typescript do 'writr init --typescript'
29
49
 
30
- ## 3. Create your first post
31
-
32
- > writr new
33
-
34
- This will allow you to answer a couple questions to setup your first blog post.
35
-
36
- ## 4. Run Writr on it with defaults. This will output everything to ./blog_output
37
-
38
- > writr
39
-
40
- ## 5. You can serve your blog with a simple webserver
50
+ ## 2. Add your content
41
51
 
42
- > writr serve
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 writr will copy it over for you automatically.
43
53
 
44
- ## 4. Integrate your blog with Express
45
-
46
- Then in express map your `blog_output` via static files:
47
-
48
- ```javascript
49
- app.use("/blog/*/images", express.static(path.join(__dirname, "blog_output/images")))
50
- app.use("/blog/images", express.static(path.join(__dirname, "blog_output/images")))
51
- app.use("/blog", express.static(path.join(__dirname, "blog_output")))
52
- ```
53
-
54
- ---
54
+ ## 3. Build your site
55
55
 
56
- ## CLI
56
+ > npx writr
57
57
 
58
- * -h, --help: Output usage information
59
- * -p, --path: Path of where the blog, and config are located
60
- * -o, --output: Path of where to output the generated blog
61
- * -r, --render: What do you want rendered such as html or json (example --render html,json)
62
- * -c, --config: Configuration file location if different than 'path'
63
- * init: Initialize a new blog
64
- * new: Create a new blog post
65
- * serve: Serve the blog. You can also specify a port with --port and --watch for hot reloading
58
+ This will build your site and place it in the `dist` folder. You can then host it anywhere you like.
66
59
 
67
- ## Templates
60
+ ## Using Your own Template
68
61
 
69
- There are three templates that are part of every instance of Writr. By default it goes in the `/blog/templates` directory. Here are the files and are in `Handlebars` format:
70
- * index.hjs: This is the main template that lists all of the latest blogs or what you want to add.
71
- * post.hjs: The post itself and usually supporting items around that such as what is next to look at and tags.
72
- * tag.hjs: Showing articles by tag filtering.
62
+ If you want to use your own template you can do so by adding a `writr.config.ts` file to the root of your project. This file will be used to configure the build process.
73
63
 
74
- ## Template Partials
64
+ or at the command line:
75
65
 
76
- You can use template partials such as a header or footer by creating a folder in templates called `partials`. In there create a standard handlebars template file such as `header.hjs`. To reference it go to any of the main template files and include it like `{{> header}}`:
66
+ > npx writr --template path/to/template
77
67
 
78
- ```html
79
- <h1>Post</h1>
68
+ ## Building Multiple Pages
80
69
 
81
- {{> header}}
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:
82
71
 
83
- <p>{{post.title}}</p>
84
- <p>{{post.author}}</p>
85
- <p>{{{post.body}}}</p>
86
-
87
- <p>{{post.matter.featured_image}}</p>
88
-
89
- <p>{{previousPost.id}}</p>
90
- <p>{{nextPost.id}}</p>
91
- ...
72
+ ```
73
+ site
74
+ ├───site.css
75
+ ├───logo.png
76
+ ├───favicon.ico
77
+ ├───writr.config.ts
78
+ ├───docs
79
+ │ ├───getting-started.md
80
+ │ ├───contributing.md
81
+ │ ├───license.md
82
+ │ ├───code-of-conduct.md
92
83
  ```
93
84
 
94
- ## Different Templates / Layouts
95
-
96
- You can also set a post to use a different layout by setting the `layout` value in the `front-matter` like so:
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:
97
86
 
98
- ```yaml
87
+ ```md
99
88
  ---
100
- title: 'Docula: Persistent Links and Styles!'
101
- tags:
102
- - Github
103
- - Open Source
104
- - Docula
105
- date: 2017-03-07
106
- layout: post2
107
- featured_image: Docula_%20Persistent%20Links%20and%20Styles%201.jpeg
89
+ title: Getting Started
90
+ order: 2
108
91
  ---
109
92
  ```
110
93
 
111
- ## Permalinks
112
-
113
- In your posts `front-matter` you can specify the format of the url to be generated. By default is the `:title` (also known as the `none` style) that is formatted correctly.
94
+ ## Helper Functions for Markdown
114
95
 
115
- ### Variables
96
+ Writr comes with some helper functions that you can use in your markdown files.
97
+ * `writrHelpers.getFrontMatter(fileName)` - Gets the front matter of a markdown file.
98
+ * `writrHelpers.setFrontMatter(fileName, frontMatter)` - Sets the front matter of a markdown file.
99
+ * `writrHelpers.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.
116
100
 
117
- | Variable | Description |
118
- | --- | ----------- |
119
- | year | Year from the post’s filename with four digits. |
120
- | short_year | Year from the post’s filename without the century. (00..99) |
121
- | month | Month from the post’s filename. (01..12) |
122
- | i_month | Month without leading zeros |
123
- | short_month | Three-letter month abbreviation, e.g. "Dec". |
124
- | long_month | Full month name, e.g. “January”. |
125
- | day | Day of the month from the post’s filename. (01..31) |
126
- | i_day | Day of the month without leading zeros from the post’s filename. |
127
- | y_day | Day of the year (01...365) |
128
- | short_day | Three-letter weekday abbreviation, e.g. “Sun”. |
129
- | long_day | Weekday name, e.g. “Sunday”. |
130
- | week | Week number of the current year, starting with the first week having a majority of its days in January. (01..53) |
131
- | hour | Hour of the day, 24-hour clock, zero-padded from the post’s date front matter. (00..23) |
132
- | minute | Minute of the hour from the post’s date front matter. (00..59) |
133
- | second | Second of the minute from the post’s date front matter. (00..59) |
134
- | title | Title from the document’s front matter. |
101
+ ### Remove html content
135
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:
136
104
 
137
- ### Default Styles
105
+ ### Get and Set the Front Matter of a Markdown File
138
106
 
139
- You can simply put in the style on permalink setting in the individual post `front-matter` or globally in `config.json`
107
+ You can use the `writrHelpers.getFrontMatter()` and `writrHelpers.setFrontMatter()` helper functions to get and set the front matter of a markdown file. Here is an example:
140
108
 
141
- | Style | Template |
142
- | --- | ----------- |
143
- | default | /:title/ |
144
- | date | /:year/:month/:day/:title/ |
145
- | ordinal | /:year/:y_day/:title/ |
109
+ ```js
110
+ const frontMatter = writrHelpers.getFrontMatter('../readme.md');
111
+ frontMatter.title = 'My Title';
112
+ writrHelpers.setFrontMatter('../readme.md', frontMatter);
113
+ ```
146
114
 
115
+ ## Code of Conduct and Contributing
116
+ [Code of Conduct](CODE_OF_CONDUCT.md) and [Contributing](CONTRIBUTING.md) guidelines.
147
117
 
148
- #### Set Layout it in the Post
118
+ ## What Happened to it Generating a Blog
149
119
 
150
- ```yaml
151
- ---
152
- title: 'Docula: Persistent Links and Styles!'
153
- tags:
154
- - Github
155
- - Open Source
156
- - Docula
157
- permalink: date
158
- date: 2017-03-07
159
- layout: post2
160
- featured_image: Docula_%20Persistent%20Links%20and%20Styles%201.jpeg
161
- ---
162
- ```
120
+ The original version of writr was a blog generator. Since there are plenty of blog generators out there we made the decision to make it a static site generator for open source projects. This is something that we constantly need and we hope you find it useful as well.
163
121
 
164
- The url will be: `/2017/03/07/docula-persistent-links-and-styles`
165
-
166
- #### Set Layout Globally
167
-
168
- To set it globally you can set it in the `config.json` by setting the `permaLink` variable like so:
169
- ```javascript
170
- {
171
- "output" : "./blog_output",
172
- "render": [ "html" , "json", "atom", "images"],
173
- "path": "./blog_example",
174
- "title": "Example Blog",
175
- "url": "https://writr.io/blog",
176
- "authorName": "Jared Wray",
177
- "authorEmail": "me@jaredwray.com",
178
- "permalink": ":year/:month/:title"
179
- }
180
- ```
122
+ ## License
181
123
 
182
- ## Markdown
183
- To learn more about Markdown go here: https://markdownguide.org
124
+ MIT © [Jared Wray](https://jaredwray.com)
package/bin/writr.mjs ADDED
@@ -0,0 +1,8 @@
1
+ #!/usr/bin/env ts-node
2
+
3
+ import * as process from 'node:process';
4
+ import Writr from '../dist/writr.js';
5
+
6
+ const writr = new Writr();
7
+
8
+ await writr.execute(process);
@@ -0,0 +1,33 @@
1
+ import { WritrOptions } from './options.js';
2
+ import { type GithubData } from './github.js';
3
+ export type WritrData = {
4
+ siteUrl: string;
5
+ siteTitle: string;
6
+ siteDescription: string;
7
+ sitePath: string;
8
+ templatePath: string;
9
+ outputPath: string;
10
+ github?: GithubData;
11
+ templates?: WritrTemplates;
12
+ };
13
+ export type WritrTemplates = {
14
+ index: string;
15
+ releases: string;
16
+ };
17
+ export declare class WritrBuilder {
18
+ private readonly _options;
19
+ private readonly _ecto;
20
+ private readonly _console;
21
+ constructor(options?: WritrOptions, engineOptions?: any);
22
+ get options(): WritrOptions;
23
+ build(): Promise<void>;
24
+ validateOptions(options: WritrOptions): void;
25
+ getGithubData(githubPath: string): Promise<GithubData>;
26
+ getTemplates(options: WritrOptions): Promise<WritrTemplates>;
27
+ getTemplateFile(path: string, name: string): Promise<string | undefined>;
28
+ buildRobotsPage(options: WritrOptions): Promise<void>;
29
+ buildSiteMapPage(data: WritrData): Promise<void>;
30
+ buildIndexPage(data: WritrData): Promise<void>;
31
+ buildReleasePage(data: WritrData): Promise<void>;
32
+ buildReadmeSection(data: WritrData): Promise<string>;
33
+ }
@@ -0,0 +1,181 @@
1
+ import { Ecto } from 'ecto';
2
+ import fs from 'fs-extra';
3
+ import { WritrOptions } from './options.js';
4
+ import { WritrConsole } from './console.js';
5
+ import { Github } from './github.js';
6
+ export class WritrBuilder {
7
+ _options = new WritrOptions();
8
+ _ecto;
9
+ _console = new WritrConsole();
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 writrData = {
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
+ };
33
+ // Get data from github
34
+ const githubData = await this.getGithubData(this.options.githubPath);
35
+ // Get data of the site
36
+ writrData.github = githubData;
37
+ // Get the templates to use
38
+ writrData.templates = await this.getTemplates(this.options);
39
+ // Build the home page (index.html)
40
+ await this.buildIndexPage(writrData);
41
+ // Build the releases page (/releases/index.html)
42
+ await this.buildReleasePage(writrData);
43
+ // Build the sitemap (/sitemap.xml)
44
+ await this.buildSiteMapPage(writrData);
45
+ // Build the robots.txt (/robots.txt)
46
+ await this.buildRobotsPage(this.options);
47
+ const siteRelativePath = this.options.sitePath;
48
+ // Copy over favicon
49
+ if (await fs.pathExists(`${siteRelativePath}/favicon.ico`)) {
50
+ await fs.copy(`${siteRelativePath}/favicon.ico`, `${this.options.outputPath}/favicon.ico`);
51
+ }
52
+ // Copy over logo
53
+ if (await fs.pathExists(`${siteRelativePath}/logo.svg`)) {
54
+ await fs.copy(`${siteRelativePath}/logo.svg`, `${this.options.outputPath}/logo.svg`);
55
+ }
56
+ // Copy over css
57
+ if (await fs.pathExists(`${this.options.templatePath}/css`)) {
58
+ await fs.copy(`${this.options.templatePath}/css`, `${this.options.outputPath}/css`);
59
+ }
60
+ // Copy over variables
61
+ if (await fs.pathExists(`${siteRelativePath}/variables.css`)) {
62
+ await fs.copy(`${siteRelativePath}/variables.css`, `${this.options.outputPath}/css/variables.css`);
63
+ }
64
+ const endTime = Date.now();
65
+ const executionTime = endTime - startTime;
66
+ this._console.log(`Build completed in ${executionTime}ms`);
67
+ }
68
+ validateOptions(options) {
69
+ if (options.githubPath.length < 3) {
70
+ throw new Error('No github options provided');
71
+ }
72
+ if (options.siteDescription.length < 3) {
73
+ throw new Error('No site description options provided');
74
+ }
75
+ if (!options.siteTitle) {
76
+ throw new Error('No site title options provided');
77
+ }
78
+ if (!options.siteUrl) {
79
+ throw new Error('No site url options provided');
80
+ }
81
+ }
82
+ async getGithubData(githubPath) {
83
+ const paths = githubPath.split('/');
84
+ const options = {
85
+ author: paths[0],
86
+ repo: paths[1],
87
+ };
88
+ const github = new Github(options);
89
+ return github.getData();
90
+ }
91
+ async getTemplates(options) {
92
+ const templates = {
93
+ index: '',
94
+ releases: '',
95
+ };
96
+ if (await fs.pathExists(options.templatePath)) {
97
+ const index = await this.getTemplateFile(options.templatePath, 'index');
98
+ if (index) {
99
+ templates.index = index;
100
+ }
101
+ const releases = await this.getTemplateFile(options.templatePath, 'releases');
102
+ if (releases) {
103
+ templates.releases = releases;
104
+ }
105
+ }
106
+ else {
107
+ throw new Error('No template path found');
108
+ }
109
+ return templates;
110
+ }
111
+ async getTemplateFile(path, name) {
112
+ let result;
113
+ const files = await fs.readdir(path);
114
+ for (const file of files) {
115
+ const fileName = file.split('.');
116
+ if (fileName[0].toString().toLowerCase() === name.toLowerCase()) {
117
+ result = file.toString();
118
+ break;
119
+ }
120
+ }
121
+ return result;
122
+ }
123
+ async buildRobotsPage(options) {
124
+ const { sitePath } = options;
125
+ const { outputPath } = options;
126
+ const robotsPath = `${outputPath}/robots.txt`;
127
+ await fs.ensureDir(outputPath);
128
+ await ((await fs.pathExists(`${sitePath}/robots.txt`))
129
+ ? fs.copy(`${sitePath}/robots.txt`, robotsPath)
130
+ : fs.writeFile(robotsPath, 'User-agent: *\nDisallow:'));
131
+ }
132
+ async buildSiteMapPage(data) {
133
+ const sitemapPath = `${data.outputPath}/sitemap.xml`;
134
+ const urls = [{ url: data.siteUrl }, { url: `${data.siteUrl}/releases` }];
135
+ let xml = '<?xml version="1.0" encoding="UTF-8"?>';
136
+ xml += '<urlset xmlns="http://www.sitemaps.org/schemas/sitemap/0.9">';
137
+ for (const { url } of urls) {
138
+ xml += '<url>';
139
+ xml += `<loc>${url}</loc>`;
140
+ xml += '</url>';
141
+ }
142
+ xml += '</urlset>';
143
+ await fs.ensureDir(data.outputPath);
144
+ await fs.writeFile(sitemapPath, xml, 'utf8');
145
+ }
146
+ async buildIndexPage(data) {
147
+ if (data.templates) {
148
+ const indexPath = `${data.outputPath}/index.html`;
149
+ await fs.ensureDir(data.outputPath);
150
+ const indexTemplate = `${data.templatePath}/${data.templates.index}`;
151
+ const htmlReadme = await this.buildReadmeSection(data);
152
+ const indexContent = await this._ecto.renderFromFile(indexTemplate, { ...data, content: htmlReadme }, data.templatePath);
153
+ await fs.writeFile(indexPath, indexContent, 'utf8');
154
+ }
155
+ else {
156
+ throw new Error('No templates found');
157
+ }
158
+ }
159
+ async buildReleasePage(data) {
160
+ if (data.github && data.templates) {
161
+ const releasesPath = `${data.outputPath}/releases/index.html`;
162
+ const releaseOutputPath = `${data.outputPath}/releases`;
163
+ await fs.ensureDir(releaseOutputPath);
164
+ const releasesTemplate = `${data.templatePath}/${data.templates.releases}`;
165
+ const releasesContent = await this._ecto.renderFromFile(releasesTemplate, data, data.templatePath);
166
+ await fs.writeFile(releasesPath, releasesContent, 'utf8');
167
+ }
168
+ else {
169
+ throw new Error('No github data found');
170
+ }
171
+ }
172
+ async buildReadmeSection(data) {
173
+ let htmlReadme = '';
174
+ if (fs.existsSync(`${data.sitePath}/README.md`)) {
175
+ const readmeContent = fs.readFileSync(`${data.sitePath}/README.md`, 'utf8');
176
+ htmlReadme = await this._ecto.markdown.render(readmeContent);
177
+ }
178
+ return htmlReadme;
179
+ }
180
+ }
181
+ //# 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 {WritrOptions} from './options.js';\nimport {WritrConsole} from './console.js';\nimport {Github, type GithubData, type GithubOptions} from './github.js';\n\nexport type WritrData = {\n\tsiteUrl: string;\n\tsiteTitle: string;\n\tsiteDescription: string;\n\tsitePath: string;\n\ttemplatePath: string;\n\toutputPath: string;\n\tgithub?: GithubData;\n\ttemplates?: WritrTemplates;\n};\n\nexport type WritrTemplates = {\n\tindex: string;\n\treleases: string;\n};\n\nexport class WritrBuilder {\n\tprivate readonly _options: WritrOptions = new WritrOptions();\n\tprivate readonly _ecto: Ecto;\n\tprivate readonly _console: WritrConsole = new WritrConsole();\n\n\tconstructor(options?: WritrOptions, 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(): WritrOptions {\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 writrData: WritrData = {\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};\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\twritrData.github = githubData;\n\t\t// Get the templates to use\n\t\twritrData.templates = await this.getTemplates(this.options);\n\n\t\t// Build the home page (index.html)\n\t\tawait this.buildIndexPage(writrData);\n\n\t\t// Build the releases page (/releases/index.html)\n\t\tawait this.buildReleasePage(writrData);\n\n\t\t// Build the sitemap (/sitemap.xml)\n\t\tawait this.buildSiteMapPage(writrData);\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: WritrOptions): 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: WritrOptions): Promise<WritrTemplates> {\n\t\tconst templates: WritrTemplates = {\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');\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: WritrOptions): 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: WritrData): 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: WritrData): 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: WritrData): 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: WritrData): 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 WritrConsole {
2
+ log(message: string): void;
3
+ error(message: string): void;
4
+ warn(message: string): void;
5
+ printHelp(): void;
6
+ parseProcessArgv(argv: string[]): WritrConsoleProcess;
7
+ getCommand(argv: string[]): string | undefined;
8
+ getArguments(argv: string[]): WritrConsoleArguments;
9
+ }
10
+ type WritrConsoleProcess = {
11
+ argv: string[];
12
+ command: string | undefined;
13
+ args: WritrConsoleArguments;
14
+ };
15
+ type WritrConsoleArguments = {
16
+ sitePath: string | undefined;
17
+ templatePath: string | undefined;
18
+ output: string | undefined;
19
+ watch: boolean;
20
+ port: number;
21
+ };
22
+ export {};