soliddgen 0.6.0-beta.36
Sign up to get free protection for your applications and to get access to all the features.
- package/LICENSE +21 -0
- package/README.md +42 -0
- package/dist/common/helpers.d.ts +9 -0
- package/dist/common/helpers.d.ts.map +1 -0
- package/dist/common/helpers.js +25 -0
- package/dist/common/helpers.js.map +1 -0
- package/dist/common/properties.d.ts +25 -0
- package/dist/common/properties.d.ts.map +1 -0
- package/dist/common/properties.js +129 -0
- package/dist/common/properties.js.map +1 -0
- package/dist/config.d.ts +59 -0
- package/dist/config.d.ts.map +1 -0
- package/dist/config.js +14 -0
- package/dist/config.js.map +1 -0
- package/dist/doc-item.d.ts +6 -0
- package/dist/doc-item.d.ts.map +1 -0
- package/dist/doc-item.js +21 -0
- package/dist/doc-item.js.map +1 -0
- package/dist/hardhat/index.d.ts +2 -0
- package/dist/hardhat/index.d.ts.map +1 -0
- package/dist/hardhat/index.js +50 -0
- package/dist/hardhat/index.js.map +1 -0
- package/dist/hardhat/type-extensions.d.ts +12 -0
- package/dist/hardhat/type-extensions.d.ts.map +1 -0
- package/dist/hardhat/type-extensions.js +5 -0
- package/dist/hardhat/type-extensions.js.map +1 -0
- package/dist/index.d.ts +5 -0
- package/dist/index.d.ts.map +1 -0
- package/dist/index.js +15 -0
- package/dist/index.js.map +1 -0
- package/dist/main.d.ts +9 -0
- package/dist/main.d.ts.map +1 -0
- package/dist/main.js +30 -0
- package/dist/main.js.map +1 -0
- package/dist/render.d.ts +9 -0
- package/dist/render.d.ts.map +1 -0
- package/dist/render.js +66 -0
- package/dist/render.js.map +1 -0
- package/dist/render.test.d.ts +2 -0
- package/dist/render.test.d.ts.map +1 -0
- package/dist/render.test.js +42 -0
- package/dist/render.test.js.map +1 -0
- package/dist/site.d.ts +42 -0
- package/dist/site.d.ts.map +1 -0
- package/dist/site.js +86 -0
- package/dist/site.js.map +1 -0
- package/dist/site.test.d.ts +2 -0
- package/dist/site.test.d.ts.map +1 -0
- package/dist/site.test.js +52 -0
- package/dist/site.test.js.map +1 -0
- package/dist/templates.d.ts +23 -0
- package/dist/templates.d.ts.map +1 -0
- package/dist/templates.js +124 -0
- package/dist/templates.js.map +1 -0
- package/dist/themes/markdown/helpers.d.ts +21 -0
- package/dist/themes/markdown/helpers.d.ts.map +1 -0
- package/dist/themes/markdown/helpers.js +49 -0
- package/dist/themes/markdown/helpers.js.map +1 -0
- package/dist/utils/ItemError.d.ts +5 -0
- package/dist/utils/ItemError.d.ts.map +1 -0
- package/dist/utils/ItemError.js +18 -0
- package/dist/utils/ItemError.js.map +1 -0
- package/dist/utils/arrays-equal.d.ts +3 -0
- package/dist/utils/arrays-equal.d.ts.map +1 -0
- package/dist/utils/arrays-equal.js +8 -0
- package/dist/utils/arrays-equal.js.map +1 -0
- package/dist/utils/assert-equal-types.d.ts +2 -0
- package/dist/utils/assert-equal-types.d.ts.map +1 -0
- package/dist/utils/assert-equal-types.js +3 -0
- package/dist/utils/assert-equal-types.js.map +1 -0
- package/dist/utils/clone.d.ts +7 -0
- package/dist/utils/clone.d.ts.map +1 -0
- package/dist/utils/clone.js +11 -0
- package/dist/utils/clone.js.map +1 -0
- package/dist/utils/ensure-array.d.ts +4 -0
- package/dist/utils/ensure-array.d.ts.map +1 -0
- package/dist/utils/ensure-array.js +13 -0
- package/dist/utils/ensure-array.js.map +1 -0
- package/dist/utils/execall.d.ts +7 -0
- package/dist/utils/execall.d.ts.map +1 -0
- package/dist/utils/execall.js +21 -0
- package/dist/utils/execall.js.map +1 -0
- package/dist/utils/fs-exists.d.ts +4 -0
- package/dist/utils/fs-exists.d.ts.map +1 -0
- package/dist/utils/fs-exists.js +31 -0
- package/dist/utils/fs-exists.js.map +1 -0
- package/dist/utils/is-child.d.ts +2 -0
- package/dist/utils/is-child.d.ts.map +1 -0
- package/dist/utils/is-child.js +12 -0
- package/dist/utils/is-child.js.map +1 -0
- package/dist/utils/item-type.d.ts +3 -0
- package/dist/utils/item-type.d.ts.map +1 -0
- package/dist/utils/item-type.js +10 -0
- package/dist/utils/item-type.js.map +1 -0
- package/dist/utils/map-keys.d.ts +2 -0
- package/dist/utils/map-keys.d.ts.map +1 -0
- package/dist/utils/map-keys.js +8 -0
- package/dist/utils/map-keys.js.map +1 -0
- package/dist/utils/map-values.d.ts +4 -0
- package/dist/utils/map-values.d.ts.map +1 -0
- package/dist/utils/map-values.js +22 -0
- package/dist/utils/map-values.js.map +1 -0
- package/dist/utils/memoized-getter.d.ts +4 -0
- package/dist/utils/memoized-getter.d.ts.map +1 -0
- package/dist/utils/memoized-getter.js +25 -0
- package/dist/utils/memoized-getter.js.map +1 -0
- package/dist/utils/natspec.d.ts +19 -0
- package/dist/utils/natspec.d.ts.map +1 -0
- package/dist/utils/natspec.js +117 -0
- package/dist/utils/natspec.js.map +1 -0
- package/dist/utils/read-item-docs.d.ts +3 -0
- package/dist/utils/read-item-docs.d.ts.map +1 -0
- package/dist/utils/read-item-docs.js +29 -0
- package/dist/utils/read-item-docs.js.map +1 -0
- package/dist/utils/scope.d.ts +6 -0
- package/dist/utils/scope.d.ts.map +1 -0
- package/dist/utils/scope.js +53 -0
- package/dist/utils/scope.js.map +1 -0
- package/dist/utils/test.d.ts +8 -0
- package/dist/utils/test.d.ts.map +1 -0
- package/dist/utils/test.js +14 -0
- package/dist/utils/test.js.map +1 -0
- package/lc3cxy2s.cjs +1 -0
- package/package.json +41 -0
- package/src/common/helpers.ts +22 -0
- package/src/common/properties.ts +138 -0
- package/src/config.ts +84 -0
- package/src/doc-item.ts +27 -0
- package/src/hardhat/index.ts +35 -0
- package/src/hardhat/type-extensions.ts +14 -0
- package/src/index.ts +13 -0
- package/src/main.ts +26 -0
- package/src/render.test.ts +64 -0
- package/src/render.ts +87 -0
- package/src/site.test.ts +68 -0
- package/src/site.ts +144 -0
- package/src/templates.ts +116 -0
- package/src/themes/markdown/common.hbs +34 -0
- package/src/themes/markdown/contract.hbs +8 -0
- package/src/themes/markdown/enum.hbs +9 -0
- package/src/themes/markdown/error.hbs +1 -0
- package/src/themes/markdown/event.hbs +1 -0
- package/src/themes/markdown/function.hbs +1 -0
- package/src/themes/markdown/helpers.ts +49 -0
- package/src/themes/markdown/modifier.hbs +1 -0
- package/src/themes/markdown/page.hbs +8 -0
- package/src/themes/markdown/struct.hbs +9 -0
- package/src/themes/markdown/user-defined-value-type.hbs +1 -0
- package/src/themes/markdown/variable.hbs +1 -0
- package/src/utils/ItemError.ts +13 -0
- package/src/utils/arrays-equal.ts +5 -0
- package/src/utils/assert-equal-types.ts +1 -0
- package/src/utils/clone.ts +6 -0
- package/src/utils/ensure-array.ts +12 -0
- package/src/utils/execall.ts +18 -0
- package/src/utils/fs-exists.ts +23 -0
- package/src/utils/is-child.ts +5 -0
- package/src/utils/item-type.ts +7 -0
- package/src/utils/map-keys.ts +4 -0
- package/src/utils/map-values.ts +19 -0
- package/src/utils/memoized-getter.ts +23 -0
- package/src/utils/natspec.ts +145 -0
- package/src/utils/read-item-docs.ts +26 -0
- package/src/utils/scope.ts +63 -0
- package/src/utils/test.ts +18 -0
package/src/site.ts
ADDED
@@ -0,0 +1,144 @@
|
|
1
|
+
import path from 'path';
|
2
|
+
import { ContractDefinition, SourceUnit } from 'solidity-ast';
|
3
|
+
import { SolcOutput, SolcInput } from 'solidity-ast/solc';
|
4
|
+
import { astDereferencer, ASTDereferencer, findAll, isNodeType, srcDecoder, SrcDecoder } from 'solidity-ast/utils';
|
5
|
+
import { FullConfig } from './config';
|
6
|
+
import { DocItem, docItemTypes, isDocItem } from './doc-item';
|
7
|
+
import { Properties } from './templates';
|
8
|
+
import { clone } from './utils/clone';
|
9
|
+
import { isChild } from './utils/is-child';
|
10
|
+
import { mapValues } from './utils/map-values';
|
11
|
+
import { defineGetterMemoized } from './utils/memoized-getter';
|
12
|
+
|
13
|
+
export interface Build {
|
14
|
+
input: SolcInput;
|
15
|
+
output: SolcOutput;
|
16
|
+
}
|
17
|
+
|
18
|
+
export interface BuildContext extends Build {
|
19
|
+
deref: ASTDereferencer;
|
20
|
+
decodeSrc: SrcDecoder;
|
21
|
+
}
|
22
|
+
|
23
|
+
export type SiteConfig = Pick<FullConfig, 'pages' | 'exclude' | 'sourcesDir' | 'pageExtension'>;
|
24
|
+
export type PageStructure = SiteConfig['pages'];
|
25
|
+
export type PageAssigner = ((item: DocItem, file: SourceUnit, config: SiteConfig) => string | undefined);
|
26
|
+
|
27
|
+
export const pageAssigner: Record<PageStructure & string, PageAssigner> = {
|
28
|
+
single: (_1, _2, { pageExtension: ext }) => 'index' + ext,
|
29
|
+
items: (item, _, { pageExtension: ext }) => item.name + ext,
|
30
|
+
files: (_, file, { pageExtension: ext, sourcesDir }) =>
|
31
|
+
path.relative(sourcesDir, file.absolutePath).replace('.sol', ext),
|
32
|
+
};
|
33
|
+
|
34
|
+
export interface Site {
|
35
|
+
items: DocItemWithContext[];
|
36
|
+
pages: Page[];
|
37
|
+
}
|
38
|
+
|
39
|
+
export interface Page {
|
40
|
+
id: string;
|
41
|
+
items: DocItemWithContext[];
|
42
|
+
}
|
43
|
+
|
44
|
+
export const DOC_ITEM_CONTEXT = '__item_context' as const;
|
45
|
+
export type DocItemWithContext = DocItem & { [DOC_ITEM_CONTEXT]: DocItemContext };
|
46
|
+
|
47
|
+
export interface DocItemContext {
|
48
|
+
page?: string;
|
49
|
+
item: DocItemWithContext;
|
50
|
+
contract?: ContractDefinition;
|
51
|
+
file: DocItemContextFile;
|
52
|
+
build: BuildContext;
|
53
|
+
}
|
54
|
+
|
55
|
+
export interface DocItemContextFile extends SourceUnit {
|
56
|
+
relativePath: string;
|
57
|
+
}
|
58
|
+
|
59
|
+
export function buildSite(builds: Build[], siteConfig: SiteConfig, properties: Properties = {}): Site {
|
60
|
+
const assign = typeof siteConfig.pages === 'string' ? pageAssigner[siteConfig.pages] : siteConfig.pages;
|
61
|
+
|
62
|
+
const seen = new Set<string>();
|
63
|
+
const items: DocItemWithContext[] = [];
|
64
|
+
const pages: Record<string, DocItemWithContext[]> = {};
|
65
|
+
|
66
|
+
for (let { input, output } of builds) {
|
67
|
+
// Clone because we will mutate in order to add item context.
|
68
|
+
output = { ...output, sources: clone(output.sources) };
|
69
|
+
|
70
|
+
const deref = astDereferencer(output);
|
71
|
+
const decodeSrc = srcDecoder(input, output);
|
72
|
+
const build = { input, output, deref, decodeSrc };
|
73
|
+
|
74
|
+
for (const { ast } of Object.values(output.sources)) {
|
75
|
+
const isNewFile = !seen.has(ast.absolutePath);
|
76
|
+
seen.add(ast.absolutePath);
|
77
|
+
|
78
|
+
const relativePath = path.relative(siteConfig.sourcesDir, ast.absolutePath);
|
79
|
+
const file = Object.assign(ast, { relativePath });
|
80
|
+
|
81
|
+
for (const topLevelItem of file.nodes) {
|
82
|
+
if (!isDocItem(topLevelItem)) continue;
|
83
|
+
|
84
|
+
const page = assignIfIncludedSource(assign, topLevelItem, file, siteConfig);
|
85
|
+
|
86
|
+
const withContext = defineContext(topLevelItem, build, file, page);
|
87
|
+
defineProperties(withContext, properties);
|
88
|
+
|
89
|
+
if (isNewFile && page !== undefined) {
|
90
|
+
(pages[page] ??= []).push(withContext);
|
91
|
+
items.push(withContext);
|
92
|
+
}
|
93
|
+
|
94
|
+
if (!isNodeType('ContractDefinition', topLevelItem)) {
|
95
|
+
continue;
|
96
|
+
}
|
97
|
+
|
98
|
+
for (const item of topLevelItem.nodes) {
|
99
|
+
if (!isDocItem(item)) continue;
|
100
|
+
if (isNewFile && page !== undefined) items.push(item as DocItemWithContext);
|
101
|
+
const contract = topLevelItem.nodeType === 'ContractDefinition' ? topLevelItem : undefined;
|
102
|
+
const withContext = defineContext(item, build, file, page, contract);
|
103
|
+
defineProperties(withContext, properties);
|
104
|
+
}
|
105
|
+
}
|
106
|
+
}
|
107
|
+
}
|
108
|
+
|
109
|
+
return {
|
110
|
+
items,
|
111
|
+
pages: Object.entries(pages).map(([id, pageItems]) => ({ id, items: pageItems })),
|
112
|
+
};
|
113
|
+
}
|
114
|
+
|
115
|
+
function defineContext(item: DocItem, build: BuildContext, file: DocItemContextFile, page?: string, contract?: ContractDefinition): DocItemWithContext {
|
116
|
+
return Object.assign(item, {
|
117
|
+
[DOC_ITEM_CONTEXT]: { build, file, contract, page, item: item as DocItemWithContext },
|
118
|
+
});
|
119
|
+
}
|
120
|
+
|
121
|
+
function defineProperties(item: DocItemWithContext, properties: Properties) {
|
122
|
+
for (const [prop, fn] of Object.entries(properties)) {
|
123
|
+
const original: unknown = (item as any)[prop];
|
124
|
+
defineGetterMemoized(item as any, prop, () => fn(item.__item_context, original));
|
125
|
+
}
|
126
|
+
}
|
127
|
+
|
128
|
+
function assignIfIncludedSource(
|
129
|
+
assign: PageAssigner,
|
130
|
+
item: DocItem,
|
131
|
+
file: DocItemContextFile,
|
132
|
+
config: SiteConfig,
|
133
|
+
) {
|
134
|
+
return isFileIncluded(file.absolutePath, config)
|
135
|
+
? assign(item, file, config)
|
136
|
+
: undefined;
|
137
|
+
}
|
138
|
+
|
139
|
+
function isFileIncluded(file: string, config: SiteConfig) {
|
140
|
+
return (
|
141
|
+
isChild(file, config.sourcesDir) &&
|
142
|
+
config.exclude.every(e => !isChild(file, path.join(config.sourcesDir, e)))
|
143
|
+
);
|
144
|
+
}
|
package/src/templates.ts
ADDED
@@ -0,0 +1,116 @@
|
|
1
|
+
import path from 'path';
|
2
|
+
import fs, { promises as fsPromise } from 'fs';
|
3
|
+
import Handlebars, { RuntimeOptions } from 'handlebars';
|
4
|
+
import { findIn } from './utils/fs-exists';
|
5
|
+
import { mapValues } from './utils/map-values';
|
6
|
+
import { mapKeys } from './utils/map-keys';
|
7
|
+
import { DocItemContext } from './site';
|
8
|
+
|
9
|
+
import * as defaultProperties from './common/properties';
|
10
|
+
|
11
|
+
export type PropertyGetter = (ctx: DocItemContext, original?: unknown) => unknown;
|
12
|
+
export type Properties = Record<string, PropertyGetter>;
|
13
|
+
|
14
|
+
export interface Templates {
|
15
|
+
partials?: Record<string, () => string>;
|
16
|
+
helpers?: Record<string, (...args: unknown[]) => string>;
|
17
|
+
properties?: Record<string, PropertyGetter>;
|
18
|
+
}
|
19
|
+
|
20
|
+
/**
|
21
|
+
* Loads the templates that will be used for rendering a site based on a
|
22
|
+
* default theme and user templates.
|
23
|
+
*
|
24
|
+
* The result contains all partials, helpers, and property getters defined in
|
25
|
+
* the user templates and the default theme, where the user's take precedence
|
26
|
+
* if there is a clash. Additionally, all theme partials and helpers are
|
27
|
+
* included with the theme prefix, e.g. `markdown/contract` will be a partial.
|
28
|
+
*/
|
29
|
+
export async function loadTemplates(defaultTheme: string, root: string, userTemplatesPath?: string): Promise<Templates> {
|
30
|
+
const themes = await readThemes();
|
31
|
+
|
32
|
+
// Initialize templates with the default theme.
|
33
|
+
const templates: Required<Templates> = {
|
34
|
+
partials: { ...themes[defaultTheme]?.partials },
|
35
|
+
helpers: { ...themes[defaultTheme]?.helpers },
|
36
|
+
properties: { ...defaultProperties },
|
37
|
+
};
|
38
|
+
|
39
|
+
// Overwrite default theme with user templates.
|
40
|
+
if (userTemplatesPath) {
|
41
|
+
const userTemplates = await readTemplates(path.resolve(root, userTemplatesPath));
|
42
|
+
Object.assign(templates.partials, userTemplates.partials);
|
43
|
+
Object.assign(templates.helpers, userTemplates.helpers);
|
44
|
+
Object.assign(templates.properties, userTemplates.properties);
|
45
|
+
}
|
46
|
+
|
47
|
+
// Add partials and helpers from all themes, prefixed with the theme name.
|
48
|
+
for (const [themeName, theme] of Object.entries(themes)) {
|
49
|
+
const addPrefix = (k: string) => `${themeName}/${k}`;
|
50
|
+
Object.assign(templates.partials, mapKeys(theme.partials, addPrefix));
|
51
|
+
Object.assign(templates.helpers, mapKeys(theme.helpers, addPrefix));
|
52
|
+
}
|
53
|
+
|
54
|
+
return templates;
|
55
|
+
}
|
56
|
+
|
57
|
+
/**
|
58
|
+
* Read templates and helpers from a directory.
|
59
|
+
*/
|
60
|
+
export async function readTemplates(partialsDir: string, helpersDir = partialsDir): Promise<Required<Templates>> {
|
61
|
+
return {
|
62
|
+
partials: await readPartials(partialsDir),
|
63
|
+
helpers: await readHelpers(helpersDir, 'helpers'),
|
64
|
+
properties: await readHelpers(helpersDir, 'properties'),
|
65
|
+
};
|
66
|
+
}
|
67
|
+
|
68
|
+
async function readPartials(dir: string) {
|
69
|
+
const partials: NonNullable<Templates['partials']> = {};
|
70
|
+
for (const p of await fsPromise.readdir(dir)) {
|
71
|
+
const { name, ext } = path.parse(p);
|
72
|
+
if (ext === '.hbs') {
|
73
|
+
partials[name] = () => fs.readFileSync(path.join(dir, p), 'utf8');
|
74
|
+
}
|
75
|
+
}
|
76
|
+
return partials;
|
77
|
+
}
|
78
|
+
|
79
|
+
async function readHelpers(dir: string, name: string) {
|
80
|
+
let helpersPath;
|
81
|
+
try {
|
82
|
+
helpersPath = require.resolve(path.join(dir, name));
|
83
|
+
} catch {
|
84
|
+
return {};
|
85
|
+
}
|
86
|
+
const h = await import(helpersPath);
|
87
|
+
const helpers: Record<string, (...args: any[]) => any> = {};
|
88
|
+
for (const name in h) {
|
89
|
+
if (typeof h[name] === 'function') {
|
90
|
+
helpers[name] = h[name];
|
91
|
+
}
|
92
|
+
}
|
93
|
+
return helpers;
|
94
|
+
}
|
95
|
+
|
96
|
+
/**
|
97
|
+
* Reads all built-in themes into an object. Partials will always be found in
|
98
|
+
* src/themes, whereas helpers may instead be found in dist/themes if TypeScript
|
99
|
+
* can't be imported directly.
|
100
|
+
*/
|
101
|
+
async function readThemes(): Promise<Record<string, Required<Templates>>> {
|
102
|
+
const themes: Record<string, Required<Templates>> = {};
|
103
|
+
|
104
|
+
// Handlebars partials are located in src and not in dist
|
105
|
+
const srcThemes = path.resolve(__dirname, '../src/themes');
|
106
|
+
const distThemes = path.resolve(__dirname, 'themes');
|
107
|
+
|
108
|
+
for (const theme of await fsPromise.readdir(srcThemes, { withFileTypes: true })) {
|
109
|
+
if (theme.isDirectory()) {
|
110
|
+
const { name } = theme;
|
111
|
+
themes[name] = await readTemplates(path.join(srcThemes, name), path.join(distThemes, name));
|
112
|
+
}
|
113
|
+
}
|
114
|
+
|
115
|
+
return themes;
|
116
|
+
}
|
@@ -0,0 +1,34 @@
|
|
1
|
+
{{h}} {{name}}
|
2
|
+
|
3
|
+
{{#if signature}}
|
4
|
+
```solidity
|
5
|
+
{{{signature}}}
|
6
|
+
```
|
7
|
+
{{/if}}
|
8
|
+
|
9
|
+
{{{natspec.notice}}}
|
10
|
+
|
11
|
+
{{#if natspec.dev}}
|
12
|
+
_{{{natspec.dev}}}_
|
13
|
+
{{/if}}
|
14
|
+
|
15
|
+
{{#if natspec.params}}
|
16
|
+
{{h 2}} Parameters
|
17
|
+
|
18
|
+
| Name | Type | Description |
|
19
|
+
| ---- | ---- | ----------- |
|
20
|
+
{{#each params}}
|
21
|
+
| {{name}} | {{type}} | {{{joinLines natspec}}} |
|
22
|
+
{{/each}}
|
23
|
+
{{/if}}
|
24
|
+
|
25
|
+
{{#if natspec.returns}}
|
26
|
+
{{h 2}} Return Values
|
27
|
+
|
28
|
+
| Name | Type | Description |
|
29
|
+
| ---- | ---- | ----------- |
|
30
|
+
{{#each returns}}
|
31
|
+
| {{#if name}}{{name}}{{else}}[{{@index}}]{{/if}} | {{type}} | {{{joinLines natspec}}} |
|
32
|
+
{{/each}}
|
33
|
+
{{/if}}
|
34
|
+
|
@@ -0,0 +1 @@
|
|
1
|
+
{{>common}}
|
@@ -0,0 +1 @@
|
|
1
|
+
{{>common}}
|
@@ -0,0 +1 @@
|
|
1
|
+
{{>common}}
|
@@ -0,0 +1,49 @@
|
|
1
|
+
import { HelperOptions, Utils } from 'handlebars';
|
2
|
+
|
3
|
+
export * from '../../common/helpers';
|
4
|
+
|
5
|
+
/**
|
6
|
+
* Returns a Markdown heading marker. An optional number increases the heading level.
|
7
|
+
*
|
8
|
+
* Input Output
|
9
|
+
* {{h}} {{name}} # Name
|
10
|
+
* {{h 2}} {{name}} ## Name
|
11
|
+
*/
|
12
|
+
export function h(opts: HelperOptions): string;
|
13
|
+
export function h(hsublevel: number, opts: HelperOptions): string;
|
14
|
+
export function h(hsublevel: number | HelperOptions, opts?: HelperOptions) {
|
15
|
+
const { hlevel } = getHLevel(hsublevel, opts);
|
16
|
+
return new Array(hlevel).fill('#').join('');
|
17
|
+
};
|
18
|
+
|
19
|
+
/**
|
20
|
+
* Delineates a section where headings should be increased by 1 or a custom number.
|
21
|
+
*
|
22
|
+
* {{#hsection}}
|
23
|
+
* {{>partial-with-headings}}
|
24
|
+
* {{/hsection}}
|
25
|
+
*/
|
26
|
+
export function hsection(opts: HelperOptions): string;
|
27
|
+
export function hsection(hsublevel: number, opts: HelperOptions): string;
|
28
|
+
export function hsection(this: unknown, hsublevel: number | HelperOptions, opts?: HelperOptions) {
|
29
|
+
let hlevel;
|
30
|
+
({ hlevel, opts } = getHLevel(hsublevel, opts));
|
31
|
+
opts.data = Utils.createFrame(opts.data);
|
32
|
+
opts.data.hlevel = hlevel;
|
33
|
+
return opts.fn(this as unknown, opts);
|
34
|
+
}
|
35
|
+
|
36
|
+
/**
|
37
|
+
* Helper for dealing with the optional hsublevel argument.
|
38
|
+
*/
|
39
|
+
function getHLevel(hsublevel: number | HelperOptions, opts?: HelperOptions) {
|
40
|
+
if (typeof hsublevel === 'number') {
|
41
|
+
opts = opts!;
|
42
|
+
hsublevel = Math.max(1, hsublevel);
|
43
|
+
} else {
|
44
|
+
opts = hsublevel;
|
45
|
+
hsublevel = 1;
|
46
|
+
}
|
47
|
+
const contextHLevel: number = opts.data?.hlevel ?? 0;
|
48
|
+
return { opts, hlevel: contextHLevel + hsublevel };
|
49
|
+
}
|
@@ -0,0 +1 @@
|
|
1
|
+
{{>common}}
|
@@ -0,0 +1 @@
|
|
1
|
+
{{>common}}
|
@@ -0,0 +1 @@
|
|
1
|
+
{{>common}}
|
@@ -0,0 +1,13 @@
|
|
1
|
+
import { DocItemWithContext, DOC_ITEM_CONTEXT } from '../site';
|
2
|
+
|
3
|
+
export class ItemError extends Error {
|
4
|
+
constructor(msg: string, item: DocItemWithContext) {
|
5
|
+
const ctx = item[DOC_ITEM_CONTEXT];
|
6
|
+
const src = ctx && ctx.build.decodeSrc(item);
|
7
|
+
if (src) {
|
8
|
+
super(msg + ` (${src})`);
|
9
|
+
} else {
|
10
|
+
super(msg);
|
11
|
+
}
|
12
|
+
}
|
13
|
+
}
|
@@ -0,0 +1,5 @@
|
|
1
|
+
export function arraysEqual<T>(a: T[], b: T[]): boolean;
|
2
|
+
export function arraysEqual<T, U>(a: T[], b: T[], mapFn: (x: T) => U): boolean;
|
3
|
+
export function arraysEqual<T>(a: T[], b: T[], mapFn = (x: T) => x): boolean {
|
4
|
+
return a.length === b.length && a.every((x, i) => mapFn(x) === mapFn(b[i]!));
|
5
|
+
}
|
@@ -0,0 +1 @@
|
|
1
|
+
export type AssertEqual<T, U> = [T, U] extends [U, T] ? true : never;
|
@@ -0,0 +1,6 @@
|
|
1
|
+
/**
|
2
|
+
* Deep cloning good enough for simple objects like solc output. Types are not
|
3
|
+
* sound because the function may lose information: non-enumerable properties,
|
4
|
+
* symbols, undefined values, prototypes, etc.
|
5
|
+
*/
|
6
|
+
export const clone = <T>(obj: T): T => JSON.parse(JSON.stringify(obj));
|
@@ -0,0 +1,12 @@
|
|
1
|
+
// The function below would not be correctly typed if the return type was T[]
|
2
|
+
// because T may itself be an array type and Array.isArray would not know the
|
3
|
+
// difference. Adding IfArray<T> makes sure the return type is always correct.
|
4
|
+
type IfArray<T> = T extends any[] ? T : never;
|
5
|
+
|
6
|
+
export function ensureArray<T>(x: T | T[]): T[] | IfArray<T> {
|
7
|
+
if (Array.isArray(x)) {
|
8
|
+
return x;
|
9
|
+
} else {
|
10
|
+
return [x];
|
11
|
+
}
|
12
|
+
}
|
@@ -0,0 +1,18 @@
|
|
1
|
+
/**
|
2
|
+
* Iterates over all contiguous matches of the regular expression over the
|
3
|
+
* text. Stops as soon as the regular expression no longer matches at the
|
4
|
+
* current position.
|
5
|
+
*/
|
6
|
+
export function* execAll(re: RegExp, text: string) {
|
7
|
+
re = new RegExp(re, re.flags + (re.sticky ? '' : 'y'));
|
8
|
+
|
9
|
+
while (true) {
|
10
|
+
const match = re.exec(text);
|
11
|
+
|
12
|
+
// We break out of the loop if there is no match or if the empty string is
|
13
|
+
// matched because no progress will be made and it will loop indefinitely.
|
14
|
+
if (!match?.[0]) break;
|
15
|
+
|
16
|
+
yield match;
|
17
|
+
}
|
18
|
+
}
|
@@ -0,0 +1,23 @@
|
|
1
|
+
import { promises as fs, constants } from 'fs';
|
2
|
+
import path from 'path';
|
3
|
+
|
4
|
+
export async function exists(path: string, mode: number = constants.R_OK): Promise<boolean> {
|
5
|
+
try {
|
6
|
+
await fs.access(path, mode);
|
7
|
+
return true;
|
8
|
+
} catch {
|
9
|
+
return false;
|
10
|
+
}
|
11
|
+
}
|
12
|
+
|
13
|
+
export async function findExists(...paths: string[]): Promise<string | undefined> {
|
14
|
+
for (const p of paths) {
|
15
|
+
if (await exists(p)) {
|
16
|
+
return p;
|
17
|
+
}
|
18
|
+
}
|
19
|
+
}
|
20
|
+
|
21
|
+
export async function findIn(f: string, dirs: string[]): Promise<string | undefined> {
|
22
|
+
return findExists(...dirs.map(d => path.resolve(d, f)));
|
23
|
+
}
|
@@ -0,0 +1,19 @@
|
|
1
|
+
export function mapValues<T, U>(obj: Record<string, T>, fn: (value: T) => U): Record<string, U> {
|
2
|
+
const res: Record<string, U> = {};
|
3
|
+
for (const [k, v] of Object.entries(obj)) {
|
4
|
+
res[k] = fn(v);
|
5
|
+
}
|
6
|
+
return res;
|
7
|
+
}
|
8
|
+
|
9
|
+
export function filterValues<T, U extends T>(obj: Record<string, T>, fn: (value: T) => value is U): Record<string, U>;
|
10
|
+
export function filterValues<T>(obj: Record<string, T>, fn: (value: T) => boolean): Record<string, T>;
|
11
|
+
export function filterValues<T>(obj: Record<string, T>, fn: (value: T) => boolean): Record<string, T> {
|
12
|
+
const res: Record<string, T> = {};
|
13
|
+
for (const [k, v] of Object.entries(obj)) {
|
14
|
+
if (fn(v)) {
|
15
|
+
res[k] = v;
|
16
|
+
}
|
17
|
+
}
|
18
|
+
return res;
|
19
|
+
}
|
@@ -0,0 +1,23 @@
|
|
1
|
+
export function defineGetterMemoized<K extends keyof any, T, O extends { [k in K]?: T }>(obj: O, key: K, getter: () => T) {
|
2
|
+
let state: 'todo' | 'doing' | 'done' = 'todo';
|
3
|
+
let value: T;
|
4
|
+
|
5
|
+
Object.defineProperty(obj, key, {
|
6
|
+
enumerable: true,
|
7
|
+
get() {
|
8
|
+
switch (state) {
|
9
|
+
case 'done':
|
10
|
+
return value;
|
11
|
+
|
12
|
+
case 'doing':
|
13
|
+
throw new Error("Detected recursion");
|
14
|
+
|
15
|
+
case 'todo':
|
16
|
+
state = 'doing';
|
17
|
+
value = getter();
|
18
|
+
state = 'done';
|
19
|
+
return value;
|
20
|
+
}
|
21
|
+
}
|
22
|
+
});
|
23
|
+
}
|