Skip to content

Commit 2981853

Browse files
Migrate custom v3 config to local plugins if needed
1 parent f494781 commit 2981853

File tree

8 files changed

+403
-77
lines changed

8 files changed

+403
-77
lines changed

app/config/import.ts

+4-55
Original file line numberDiff line numberDiff line change
@@ -1,64 +1,13 @@
1-
import {copySync, existsSync, writeFileSync, readFileSync, copy} from 'fs-extra';
1+
import {readFileSync} from 'fs-extra';
22
import {sync as mkdirpSync} from 'mkdirp';
3-
import {
4-
defaultCfg,
5-
cfgPath,
6-
legacyCfgPath,
7-
plugs,
8-
defaultPlatformKeyPath,
9-
schemaPath,
10-
cfgDir,
11-
schemaFile
12-
} from './paths';
13-
import {_init, _extractDefault} from './init';
3+
import {defaultCfg, cfgPath, plugs, defaultPlatformKeyPath} from './paths';
4+
import {_init} from './init';
145
import notify from '../notify';
156
import {rawConfig} from '../../lib/config';
16-
import _ from 'lodash';
17-
import {resolve} from 'path';
7+
import {migrateHyper3Config} from './migrate';
188

199
let defaultConfig: rawConfig;
2010

21-
const _write = (path: string, data: string) => {
22-
// This method will take text formatted as Unix line endings and transform it
23-
// to text formatted with DOS line endings. We do this because the default
24-
// text editor on Windows (notepad) doesn't Deal with LF files. Still. In 2017.
25-
const crlfify = (str: string) => {
26-
return str.replace(/\r?\n/g, '\r\n');
27-
};
28-
const format = process.platform === 'win32' ? crlfify(data.toString()) : data;
29-
writeFileSync(path, format, 'utf8');
30-
};
31-
32-
// Migrate Hyper3 config to Hyper4 but only if the user hasn't manually
33-
// touched the new config and if the old config is not a symlink
34-
const migrateHyper3Config = () => {
35-
copy(schemaPath, resolve(cfgDir, schemaFile), (err) => {
36-
if (err) {
37-
console.error(err);
38-
}
39-
});
40-
41-
if (existsSync(cfgPath)) {
42-
return;
43-
}
44-
45-
if (!existsSync(legacyCfgPath)) {
46-
copySync(defaultCfg, cfgPath);
47-
return;
48-
}
49-
50-
// Migrate
51-
const defaultCfgData = JSON.parse(readFileSync(defaultCfg, 'utf8'));
52-
const legacyCfgData = _extractDefault(readFileSync(legacyCfgPath, 'utf8'));
53-
const newCfgData = _.merge(defaultCfgData, legacyCfgData);
54-
_write(cfgPath, JSON.stringify(newCfgData, null, 2));
55-
56-
notify(
57-
'Hyper 4',
58-
`Settings location and format has changed.\nWe've automatically migrated your existing config to ${cfgPath}`
59-
);
60-
};
61-
6211
const _importConf = () => {
6312
// init plugin directories if not present
6413
mkdirpSync(plugs.base);

app/config/init.ts

+1-1
Original file line numberDiff line numberDiff line change
@@ -32,7 +32,7 @@ const _init = (userCfg: rawConfig, defaultCfg: rawConfig): parsedConfig => {
3232
return {
3333
config: (() => {
3434
if (userCfg?.config) {
35-
return _.merge(defaultCfg.config, userCfg.config);
35+
return _.merge({}, defaultCfg.config, userCfg.config);
3636
} else {
3737
notify('Error reading configuration: `config` key is missing');
3838
return defaultCfg.config || ({} as configOptions);

app/config/migrate.ts

+187
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,187 @@
1+
import {parse, prettyPrint} from 'recast';
2+
import {builders, namedTypes} from 'ast-types';
3+
import * as babelParser from 'recast/parsers/babel';
4+
import {copy, copySync, existsSync, readFileSync, writeFileSync} from 'fs-extra';
5+
import {dirname, resolve} from 'path';
6+
import _ from 'lodash';
7+
8+
import notify from '../notify';
9+
import {_extractDefault} from './init';
10+
import {cfgDir, cfgPath, defaultCfg, legacyCfgPath, plugs, schemaFile, schemaPath} from './paths';
11+
12+
// function to remove all json serializable entries from an array expression
13+
function removeElements(node: namedTypes.ArrayExpression): namedTypes.ArrayExpression {
14+
const newElements = node.elements.filter((element) => {
15+
if (namedTypes.ObjectExpression.check(element)) {
16+
const newElement = removeProperties(element);
17+
if (newElement.properties.length === 0) {
18+
return false;
19+
}
20+
} else if (namedTypes.ArrayExpression.check(element)) {
21+
const newElement = removeElements(element);
22+
if (newElement.elements.length === 0) {
23+
return false;
24+
}
25+
} else if (namedTypes.Literal.check(element)) {
26+
return false;
27+
}
28+
return true;
29+
});
30+
return {...node, elements: newElements};
31+
}
32+
33+
// function to remove all json serializable properties from an object expression
34+
function removeProperties(node: namedTypes.ObjectExpression): namedTypes.ObjectExpression {
35+
const newProperties = node.properties.filter((property) => {
36+
if (
37+
namedTypes.ObjectProperty.check(property) &&
38+
(namedTypes.Literal.check(property.key) || namedTypes.Identifier.check(property.key)) &&
39+
!property.computed
40+
) {
41+
if (namedTypes.ObjectExpression.check(property.value)) {
42+
const newValue = removeProperties(property.value);
43+
if (newValue.properties.length === 0) {
44+
return false;
45+
}
46+
} else if (namedTypes.ArrayExpression.check(property.value)) {
47+
const newValue = removeElements(property.value);
48+
if (newValue.elements.length === 0) {
49+
return false;
50+
}
51+
} else if (namedTypes.Literal.check(property.value)) {
52+
return false;
53+
}
54+
}
55+
return true;
56+
});
57+
return {...node, properties: newProperties};
58+
}
59+
60+
export function configToPlugin(code: string): string {
61+
const ast: namedTypes.File = parse(code, {
62+
parser: babelParser
63+
});
64+
const statements = ast.program.body;
65+
let moduleExportsNode: namedTypes.AssignmentExpression | null = null;
66+
let configNode: any = null;
67+
68+
for (const statement of statements) {
69+
if (namedTypes.ExpressionStatement.check(statement)) {
70+
const expression = statement.expression;
71+
if (
72+
namedTypes.AssignmentExpression.check(expression) &&
73+
expression.operator === '=' &&
74+
namedTypes.MemberExpression.check(expression.left) &&
75+
namedTypes.Identifier.check(expression.left.object) &&
76+
expression.left.object.name === 'module' &&
77+
namedTypes.Identifier.check(expression.left.property) &&
78+
expression.left.property.name === 'exports'
79+
) {
80+
moduleExportsNode = expression;
81+
if (namedTypes.ObjectExpression.check(expression.right)) {
82+
const properties = expression.right.properties;
83+
for (const property of properties) {
84+
if (
85+
namedTypes.ObjectProperty.check(property) &&
86+
namedTypes.Identifier.check(property.key) &&
87+
property.key.name === 'config'
88+
) {
89+
configNode = property.value;
90+
if (namedTypes.ObjectExpression.check(property.value)) {
91+
configNode = removeProperties(property.value);
92+
}
93+
}
94+
}
95+
} else {
96+
configNode = builders.memberExpression(moduleExportsNode.right, builders.identifier('config'));
97+
}
98+
}
99+
}
100+
}
101+
102+
if (!moduleExportsNode) {
103+
console.log('No module.exports found in config');
104+
return '';
105+
}
106+
if (!configNode) {
107+
console.log('No config field found in module.exports');
108+
return '';
109+
}
110+
if (namedTypes.ObjectExpression.check(configNode) && configNode.properties.length === 0) {
111+
return '';
112+
}
113+
114+
moduleExportsNode.right = builders.objectExpression([
115+
builders.property(
116+
'init',
117+
builders.identifier('decorateConfig'),
118+
builders.arrowFunctionExpression(
119+
[builders.identifier('_config')],
120+
builders.callExpression(
121+
builders.memberExpression(builders.identifier('Object'), builders.identifier('assign')),
122+
[builders.objectExpression([]), builders.identifier('_config'), configNode]
123+
)
124+
)
125+
)
126+
]);
127+
128+
return prettyPrint(ast, {tabWidth: 2}).code;
129+
}
130+
131+
export const _write = (path: string, data: string) => {
132+
// This method will take text formatted as Unix line endings and transform it
133+
// to text formatted with DOS line endings. We do this because the default
134+
// text editor on Windows (notepad) doesn't Deal with LF files. Still. In 2017.
135+
const crlfify = (str: string) => {
136+
return str.replace(/\r?\n/g, '\r\n');
137+
};
138+
const format = process.platform === 'win32' ? crlfify(data.toString()) : data;
139+
writeFileSync(path, format, 'utf8');
140+
};
141+
142+
// Migrate Hyper3 config to Hyper4 but only if the user hasn't manually
143+
// touched the new config and if the old config is not a symlink
144+
export const migrateHyper3Config = () => {
145+
copy(schemaPath, resolve(cfgDir, schemaFile), (err) => {
146+
if (err) {
147+
console.error(err);
148+
}
149+
});
150+
151+
if (existsSync(cfgPath)) {
152+
return;
153+
}
154+
155+
if (!existsSync(legacyCfgPath)) {
156+
copySync(defaultCfg, cfgPath);
157+
return;
158+
}
159+
160+
// Migrate
161+
copySync(resolve(dirname(legacyCfgPath), '.hyper_plugins', 'local'), plugs.local);
162+
163+
const defaultCfgData = JSON.parse(readFileSync(defaultCfg, 'utf8'));
164+
let newCfgData;
165+
try {
166+
const legacyCfgRaw = readFileSync(legacyCfgPath, 'utf8');
167+
const legacyCfgData = _extractDefault(legacyCfgRaw);
168+
newCfgData = _.merge({}, defaultCfgData, legacyCfgData);
169+
170+
const pluginCode = configToPlugin(legacyCfgRaw);
171+
if (pluginCode) {
172+
const pluginPath = resolve(plugs.local, 'migrated-hyper3-config.js');
173+
newCfgData.localPlugins = ['migrated-hyper3-config', ...(newCfgData.localPlugins || [])];
174+
_write(pluginPath, pluginCode);
175+
}
176+
} catch (e) {
177+
console.error(e);
178+
notify(
179+
'Hyper 4',
180+
`Failed to migrate your config from Hyper 3.\nDefault config will be created instead at ${cfgPath}`
181+
);
182+
newCfgData = defaultCfgData;
183+
}
184+
_write(cfgPath, JSON.stringify(newCfgData, null, 2));
185+
186+
notify('Hyper 4', `Settings location and format has changed to ${cfgPath}`);
187+
};

app/package.json

+2
Original file line numberDiff line numberDiff line change
@@ -10,6 +10,7 @@
1010
},
1111
"repository": "zeit/hyper",
1212
"dependencies": {
13+
"@babel/parser": "7.20.7",
1314
"@electron/remote": "2.0.9",
1415
"async-retry": "1.3.3",
1516
"chokidar": "^3.5.3",
@@ -31,6 +32,7 @@
3132
"queue": "6.0.2",
3233
"react": "17.0.2",
3334
"react-dom": "17.0.2",
35+
"recast": "0.22.0",
3436
"semver": "7.3.8",
3537
"shell-env": "3.0.1",
3638
"sudo-prompt": "^9.2.1",

app/plugins.ts

+5-2
Original file line numberDiff line numberDiff line change
@@ -306,9 +306,12 @@ function requirePlugins(): any[] {
306306
}
307307
};
308308

309-
return plugins_
309+
return [
310+
...localPlugins.filter((p) => basename(p) === 'migrated-hyper3-config'),
311+
...plugins_,
312+
...localPlugins.filter((p) => basename(p) !== 'migrated-hyper3-config')
313+
]
310314
.map(load)
311-
.concat(localPlugins.map(load))
312315
.filter((v) => Boolean(v));
313316
}
314317

0 commit comments

Comments
 (0)