-
-
Notifications
You must be signed in to change notification settings - Fork 5
/
Copy pathutils.js
227 lines (188 loc) · 7.31 KB
/
utils.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
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
import fetch, {FormData, fileFrom} from 'node-fetch';
import open from 'open';
import child_process from 'child_process';
import fs from 'fs-extra';
import path from 'path';
import querystring from 'querystring';
import url from 'url';
import util from 'util';
import {parseString} from 'xml2js';
function logWithInspect(object) {
// https://stackoverflow.com/a/10729284/15487978
console.log(util.inspect(object, {depth: null, colors: true}));
}
function isAppNameAllowed(appName) {
// Finder doesn't seem to appear in the dock data
const disallowedAppNames = [
'Preview',
'Launchpad'
];
return !disallowedAppNames.includes(appName);
}
function getAppNamesToIconPaths(parsedDockData) {
const parsedAppData = parsedDockData.plist.dict[0]
//logWithInspect(parsedAppData);
const persistentApps = parsedAppData.array[1].dict;
const _persistentOthers = parsedAppData.array[2].dict;
const _recentApps = parsedAppData.array[3].dict;
const result = {};
for (const parsedAppData of persistentApps ?? []) {
const appName = parsedAppData.dict[0].string[1];
const appDirectoryUrl = parsedAppData.dict[0].dict?.[0].string[0];
if (appDirectoryUrl && isAppNameAllowed(appName)) {
const appDirectory = url.fileURLToPath(appDirectoryUrl)
result[appName] = getIconPath(appDirectory);
}
}
return result;
}
/**
*
* @param appNames
* @returns {Promise<Array.<{name: string; foundInDb: boolean; missingAppIcon: boolean}>>}
*/
async function getWhichAppsAreMissingFromDatabase(appNames) {
if (!appNames.length) {
return [];
}
const queryString = querystring.stringify({app: appNames});
const url = `https://www.dockhunt.com/api/cli/check-apps?${queryString}`;
// const url = `http://localhost:3000/api/cli/check-apps?${queryString}`;
const response = await fetch(url);
if (!response.ok) {
throw 'Bad response from Dockhunt `check-apps` endpoint';
}
const payload = await response.json();
return payload.missingAppsInformation;
}
function getIconPath(appDirectory) {
var appResourcesDirectory = path.join(appDirectory, 'Contents', 'Resources');
// AppName.app/Contents/Resources may not exist for Catalyst apps.
if (!fs.pathExistsSync(appResourcesDirectory)) return null
const files = fs.readdirSync(appResourcesDirectory)
for (const file of files) {
if (file.endsWith('.icns')) {
return path.join(appResourcesDirectory, file);
}
}
return null;
}
/**
*
* @param appName {string}
* @param iconPath {string | null}
* @returns {Promise<void>}
*/
const addAppToDatabase = async (appName, iconPath) => {
const URL = 'https://dockhunt.com/api/cli/icon-upload';
// const URL = 'http://localhost:3000/api/cli/icon-upload';
const form = new FormData();
form.append('app', appName);
if (iconPath !== null) {
form.append('icon', await fileFrom(iconPath));
}
const response = await fetch(URL, {
method: 'POST',
body: form,
});
await response.json()
}
export function icns2png(appName, icnsPath, outputDir) {
return new Promise((resolve, reject) => {
const outputPath = path.join(outputDir, appName + '.png');
console.log(`Converting icon to PNG (${appName})`);
// https://stackoverflow.com/a/62892482/15487978
// https://stackoverflow.com/a/10232330/15487978
const sips = child_process.spawn('sips',
['-s', 'format', 'png', icnsPath, '--out', outputPath]
);
sips.stdout.on('data', function (data) {
//console.log('stdout: ' + data.toString());
});
sips.stderr.on('data', function (data) {
console.error('stderr: ' + data.toString());
});
sips.on('exit', function (code) {
if (!code === 0) {
console.error('child process exited with code ' + code.toString());
reject();
}
resolve({iconPath: outputPath, appName});
});
});
}
export async function scanDockAndBringToWebApp(dockXmlPlist) {
if (!dockXmlPlist.match(/<!DOCTYPE plist/g)) {
throw 'Dock data appears to be invalid. Expected: Apple plist XML.';
}
const parsedDockData = await new Promise((resolve, reject) => {
parseString(dockXmlPlist, function (error, result) {
return error ? reject(error) : resolve(result);
});
});
const appNamesToIconPaths = getAppNamesToIconPaths(parsedDockData);
const appNames = Object.keys(appNamesToIconPaths);
if (appNames.length) {
console.log('Found the following pinned apps in your dock:\n')
for (const name of appNames) {
console.log(`• ${name}`);
}
} else {
console.log('Found what appears to be an empty dock.');
}
// console.log('\nUploading missing dock icons to dockhunt...');
const appsMissingFromDatabase = await getWhichAppsAreMissingFromDatabase(
appNames
);
// Make a temporary dir for converted images
let tempDir;
if (appsMissingFromDatabase.length) {
const tempDirname = `temp_${Date.now()}_icon_conversion`;
tempDir = path.join(process.cwd(), tempDirname);
fs.mkdirSync(tempDir);
}
/** @type {Promise<{iconPath: string | null, appName: string}>[]} */
const missingAppsToBeAddedToDatabasePromises = [];
for (const app of appsMissingFromDatabase) {
const iconPath = appNamesToIconPaths[app.name];
if (!iconPath) {
console.warn(`\n${app.name} icon not found.`);
}
if (iconPath) {
missingAppsToBeAddedToDatabasePromises.push(icns2png(app.name, iconPath, tempDir));
} else if (!iconPath && !app.foundInDb) {
// We still want to upload apps to our database if they don't have an icon AND are not in our database
missingAppsToBeAddedToDatabasePromises.push(new Promise((resolve) => resolve({
iconPath: null,
appName: app.name
})));
}
}
try {
const missingAppsToBeAddedToDatabase = await Promise.all(missingAppsToBeAddedToDatabasePromises);
/** @type {Promise<void>[]} */
const appIconUploadPromises = [];
for (const app of missingAppsToBeAddedToDatabase) {
appIconUploadPromises.push(addAppToDatabase(app.appName, app.iconPath));
}
// Wait for all uploads to complete
await Promise.all(appIconUploadPromises);
// Remove temporary directory
if (tempDir) {
fs.removeSync(tempDir)
}
// Output message saying that upload is complete
console.log('\nDock scan complete!');
if (appNames.length) {
const dockhuntUrl = `https://dockhunt.com/new-dock?${appNames.map(appName => `app=${encodeURIComponent(appName)}`).join('&')}`;
// const dockhuntUrl = `http://localhost:3000/new-dock?${appNames.map(appName => `app=${encodeURIComponent(appName)}`).join('&')}`;
console.log(`\nRedirecting to dockhunt: ${dockhuntUrl}`);
await open(dockhuntUrl);
} else {
console.log('\nDockhunt does not currently support users making ' +
'Docks which contain no apps.');
}
} catch (error) {
console.error("Error converting icons to pngs:", error);
}
}