-
Notifications
You must be signed in to change notification settings - Fork 5
/
Copy pathupdate.js
executable file
·129 lines (105 loc) · 3.26 KB
/
update.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
#!/usr/bin/env node
const path = require("node:path");
const { execFileSync } = require("node:child_process");
const { mkdirSync, rmSync } = require("node:fs");
const { writeFile } = require("node:fs/promises");
const fontkit = require("fontkit");
const fetch = require("node-fetch");
const postcss = require("postcss");
const { toKebabCase } = require("./src/utils.js");
const TEMP = path.resolve(".temp");
const interUrl = "https://rsms.me/inter/";
const interSource = `${interUrl}inter.css`;
const interFamilies = new Set();
const interFiles = new Set();
const interFontFaces = new Set();
const inter = {
version: null,
availableFeatures: new Set(),
base: {
"@font-face": [],
},
utilities: {
".font-inter": {
"font-family": "Inter, sans-serif",
},
"@supports(font-variation-settings: normal)": {
".font-inter": {
"font-family": "InterVariable, sans-serif",
},
},
},
};
const extractCss = (root) => {
console.info("Parsing fetched CSS...");
root.walkAtRules((rule) => {
const declarations = {};
rule.walkDecls((decl) => {
const name = toKebabCase(decl.prop);
let value = decl.value
.split(",")
.map((val) => val.trim())
.join(", ");
if (name === "font-family") {
const fontName = value.replace(/^['"]|['"]$/g, "").trim();
if (!fontName.match(/\salt$/)) {
interFamilies.add(fontName);
}
}
if (name === "src") {
const vals = value.match(/url\(['"]([^'"]*)['"]\)/gi);
if (vals) {
for (const val of vals) {
const cleanVal = val.replace(/^url\(['"]|['"]\)$/g, "");
value = value.replace(cleanVal, interUrl + cleanVal);
interFiles.add(interUrl + cleanVal);
}
}
}
declarations[name] = value;
});
if (rule.name === 'font-face' && ['Inter', 'InterVariable'].includes(declarations['font-family'])) {
interFontFaces.add(declarations);
}
});
console.log("Found font families:", [...interFamilies].join(", "));
};
const download = (fileUrl, initialFile = null) => {
let file = initialFile;
if (!file) {
const parsed = new URL(fileUrl);
const fileName = path.basename(parsed.pathname);
file = path.join(TEMP, fileName);
}
execFileSync("curl", ["--silent", "-o", file, "-L", fileUrl], {
encoding: "utf8",
});
return file;
};
console.info("Fetching", interSource);
rmSync(TEMP, { recursive: true, force: true });
mkdirSync(TEMP, { recursive: true });
fetch(interSource)
.then((res) => res.text())
.then((css) => postcss([extractCss]).process(css, { from: undefined }))
.then(() => {
console.info("Fetching font files...");
for (const fontFile of interFiles) {
const file = download(fontFile);
const font = fontkit.openSync(file);
inter.version = font.version.replace(/Version\s/, "");
for (const feature of font.availableFeatures) {
inter.availableFeatures.add(feature);
}
}
inter.availableFeatures = [...inter.availableFeatures].sort();
inter.base['@font-face'] = [...interFontFaces];
console.log("Font version:", inter.version);
console.log("Found font features:", inter.availableFeatures.join(", "));
})
.then(async () => {
const file = path.join(process.cwd(), "inter.json");
await writeFile(file, JSON.stringify(inter));
console.info("Finished. Meta data stored in ./inter.json");
})
.catch((err) => console.error(err));