2023-05-31 10:06:00 -04:00
|
|
|
import stripIndent from 'strip-indent';
|
|
|
|
|
2023-03-09 17:32:49 -05:00
|
|
|
import { liquidEngine } from '../engines';
|
2023-03-08 15:59:19 -05:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Renders a status dashboard for each implementation's support for a feature.
|
|
|
|
*
|
|
|
|
* Each implementation's value can be:
|
|
|
|
*
|
|
|
|
* - `true`, indicating that that implementation fully supports the feature;
|
|
|
|
* - `false`, indicating that it does not yet support the feature at all;
|
2023-06-01 17:59:54 -04:00
|
|
|
* - `"partial"`, indicating that it has limited or incorrect support for the
|
2023-03-08 15:59:19 -05:00
|
|
|
* feature;
|
|
|
|
* - or a string, indicating the version it started supporting the feature.
|
|
|
|
*
|
|
|
|
* When possible, prefer using the start version rather than `true`.
|
|
|
|
*
|
|
|
|
* If `feature` is passed, it should be a terse (one- to three-word) description
|
|
|
|
* of the particular feature whose compatibility is described. This should be
|
|
|
|
* used whenever the status isn't referring to the entire feature being
|
|
|
|
* described by the surrounding prose.
|
|
|
|
*
|
|
|
|
* This takes an optional Markdown block (`details`) that should provide more
|
|
|
|
* information about the implementation differences or the old behavior.
|
|
|
|
*/
|
2023-06-01 11:43:54 -04:00
|
|
|
export const compatibility = async (details: string, ...opts: string[]) => {
|
|
|
|
const options = parseCompatibilityOpts(...opts);
|
|
|
|
return liquidEngine.renderFile('compatibility', {
|
2023-05-31 10:06:00 -04:00
|
|
|
details: stripIndent(details),
|
2023-06-01 11:43:54 -04:00
|
|
|
...options,
|
2023-03-08 15:59:19 -05:00
|
|
|
});
|
2023-06-01 11:43:54 -04:00
|
|
|
};
|
|
|
|
|
|
|
|
interface CompatibilityOptions {
|
|
|
|
dart: string | boolean | null;
|
|
|
|
libsass: string | boolean | null;
|
|
|
|
node: string | boolean | null;
|
|
|
|
ruby: string | boolean | null;
|
|
|
|
feature: string | null;
|
|
|
|
useMarkdown: boolean;
|
|
|
|
}
|
|
|
|
|
|
|
|
const extend = <
|
|
|
|
K extends keyof CompatibilityOptions,
|
|
|
|
V extends CompatibilityOptions[K],
|
|
|
|
>(
|
|
|
|
value: V,
|
|
|
|
obj: CompatibilityOptions,
|
|
|
|
key: K,
|
|
|
|
) => {
|
|
|
|
obj[key] = value;
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
2023-06-01 17:59:54 -04:00
|
|
|
* Take a list of string `args` and converts it into an object of all arguments
|
2023-06-01 11:43:54 -04:00
|
|
|
* suitable for the `compatibility.liquid` template.
|
|
|
|
*/
|
|
|
|
const parseCompatibilityOpts = (...args: string[]): CompatibilityOptions => {
|
2023-06-01 16:56:54 -04:00
|
|
|
const opts = {
|
2023-06-01 11:43:54 -04:00
|
|
|
dart: null,
|
|
|
|
libsass: null,
|
|
|
|
node: null,
|
|
|
|
ruby: null,
|
|
|
|
feature: null,
|
|
|
|
useMarkdown: true,
|
|
|
|
};
|
2023-06-01 16:56:54 -04:00
|
|
|
const keyValueRegex = /(.*?):(.*)/;
|
|
|
|
for (const arg of args) {
|
|
|
|
if (typeof arg !== 'string') {
|
2023-06-01 11:43:54 -04:00
|
|
|
throw new Error(
|
|
|
|
// eslint-disable-next-line @typescript-eslint/restrict-template-expressions
|
2023-06-01 16:56:54 -04:00
|
|
|
`Received non-string argument to {% compatibility %} tag: ${arg}`,
|
2023-06-01 11:43:54 -04:00
|
|
|
);
|
|
|
|
}
|
2023-06-01 16:56:54 -04:00
|
|
|
const match = arg.match(keyValueRegex);
|
|
|
|
if (!match) {
|
|
|
|
throw new Error(
|
|
|
|
`Arguments should be in the format 'key:value'; received ${arg}.`,
|
2023-06-01 11:43:54 -04:00
|
|
|
);
|
2023-06-01 16:56:54 -04:00
|
|
|
}
|
|
|
|
const key: string = match[1].trim();
|
|
|
|
let value: string | boolean | null = match[2].trim();
|
|
|
|
try {
|
|
|
|
// handles true, false, null, numbers, strings...
|
|
|
|
value = JSON.parse(value) as string | boolean | null;
|
|
|
|
} catch (e) {
|
|
|
|
throw new Error(
|
|
|
|
`Unable to parse argument ${key} with value ${
|
|
|
|
value as string
|
|
|
|
}. Try wrapping it in double quotes: ${key}:"${value as string}"`,
|
|
|
|
);
|
|
|
|
}
|
|
|
|
if (key && Object.hasOwn(opts, key)) {
|
|
|
|
extend(value, opts, key as keyof CompatibilityOptions);
|
2023-06-01 11:43:54 -04:00
|
|
|
} else {
|
|
|
|
throw new Error(
|
|
|
|
// eslint-disable-next-line @typescript-eslint/restrict-template-expressions
|
2023-06-01 16:56:54 -04:00
|
|
|
`Received unexpected argument to {% compatibility %} tag: ${arg}`,
|
2023-06-01 11:43:54 -04:00
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
2023-06-01 16:56:54 -04:00
|
|
|
return opts;
|
2023-06-01 11:43:54 -04:00
|
|
|
};
|
2023-03-08 15:59:19 -05:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Renders a single row for `compatibility`.
|
|
|
|
*/
|
|
|
|
export const implStatus = (status: string | boolean | null) => {
|
|
|
|
switch (status) {
|
|
|
|
case true:
|
|
|
|
return '✓';
|
|
|
|
case false:
|
|
|
|
return '✗';
|
|
|
|
case 'partial':
|
|
|
|
case null:
|
|
|
|
return status;
|
|
|
|
default:
|
|
|
|
return `since ${status}`;
|
|
|
|
}
|
|
|
|
};
|