@electron-forge/publisher-electron-release-server 6.0.1 → 6.0.2
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/LICENSE +19 -0
- package/package.json +8 -4
- package/src/Config.ts +34 -0
- package/src/PublisherERS.ts +153 -0
- package/test/PublisherERS_spec.ts +223 -0
- package/dist/Config.d.ts +0 -34
- package/dist/Config.d.ts.map +0 -1
- package/dist/Config.js +0 -6
- package/dist/PublisherERS.d.ts +0 -10
- package/dist/PublisherERS.d.ts.map +0 -1
- package/dist/PublisherERS.js +0 -150
package/LICENSE
ADDED
|
@@ -0,0 +1,19 @@
|
|
|
1
|
+
The MIT License (MIT)
|
|
2
|
+
Copyright (c) 2016 Samuel Attard
|
|
3
|
+
|
|
4
|
+
Permission is hereby granted, free of charge, to any person obtaining a copy of
|
|
5
|
+
this software and associated documentation files (the "Software"), to deal in
|
|
6
|
+
the Software without restriction, including without limitation the rights to
|
|
7
|
+
use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of
|
|
8
|
+
the Software, and to permit persons to whom the Software is furnished to do so,
|
|
9
|
+
subject to the following conditions:
|
|
10
|
+
|
|
11
|
+
The above copyright notice and this permission notice shall be included in all
|
|
12
|
+
copies or substantial portions of the Software.
|
|
13
|
+
|
|
14
|
+
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
15
|
+
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS
|
|
16
|
+
FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR
|
|
17
|
+
COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER
|
|
18
|
+
IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
|
|
19
|
+
CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@electron-forge/publisher-electron-release-server",
|
|
3
|
-
"version": "6.0.
|
|
3
|
+
"version": "6.0.2",
|
|
4
4
|
"description": "Electron release server publisher for Electron Forge",
|
|
5
5
|
"repository": "https://github.com/electron/forge",
|
|
6
6
|
"author": "Samuel Attard",
|
|
@@ -18,11 +18,15 @@
|
|
|
18
18
|
"node": ">= 14.17.5"
|
|
19
19
|
},
|
|
20
20
|
"dependencies": {
|
|
21
|
-
"@electron-forge/publisher-base": "6.0.
|
|
22
|
-
"@electron-forge/shared-types": "6.0.
|
|
21
|
+
"@electron-forge/publisher-base": "^6.0.2",
|
|
22
|
+
"@electron-forge/shared-types": "^6.0.2",
|
|
23
23
|
"debug": "^4.3.1",
|
|
24
24
|
"form-data": "^4.0.0",
|
|
25
25
|
"fs-extra": "^10.0.0",
|
|
26
26
|
"node-fetch": "^2.6.7"
|
|
27
|
-
}
|
|
27
|
+
},
|
|
28
|
+
"publishConfig": {
|
|
29
|
+
"access": "public"
|
|
30
|
+
},
|
|
31
|
+
"gitHead": "11cf4b58359c9881c05c06e0d62be575a0ed70d1"
|
|
28
32
|
}
|
package/src/Config.ts
ADDED
|
@@ -0,0 +1,34 @@
|
|
|
1
|
+
export interface PublisherERSConfig {
|
|
2
|
+
/**
|
|
3
|
+
* The base URL of your instance of ERS.
|
|
4
|
+
*
|
|
5
|
+
* E.g. https://my-update.server.com
|
|
6
|
+
*/
|
|
7
|
+
baseUrl: string;
|
|
8
|
+
/**
|
|
9
|
+
* The username you use to sign in to ERS
|
|
10
|
+
*/
|
|
11
|
+
username: string;
|
|
12
|
+
/**
|
|
13
|
+
* The password you use to sign in to ERS
|
|
14
|
+
*/
|
|
15
|
+
password: string;
|
|
16
|
+
/**
|
|
17
|
+
* The release channel you want to send artifacts to, normally something like
|
|
18
|
+
* "stable", "beta" or "alpha".
|
|
19
|
+
*
|
|
20
|
+
* If left unspecified we will try to infer the channel from your version
|
|
21
|
+
* field in your package.json.
|
|
22
|
+
*
|
|
23
|
+
* Default: stable
|
|
24
|
+
*/
|
|
25
|
+
channel?: string;
|
|
26
|
+
|
|
27
|
+
/**
|
|
28
|
+
* The "flavor" of the binary that you want to release to.
|
|
29
|
+
* This is useful if you want to provide multiple versions
|
|
30
|
+
* of the same application version (e.g. full and lite)
|
|
31
|
+
* to end users.
|
|
32
|
+
*/
|
|
33
|
+
flavor?: string;
|
|
34
|
+
}
|
|
@@ -0,0 +1,153 @@
|
|
|
1
|
+
import path from 'path';
|
|
2
|
+
|
|
3
|
+
import { PublisherBase, PublisherOptions } from '@electron-forge/publisher-base';
|
|
4
|
+
import { ForgeArch, ForgePlatform } from '@electron-forge/shared-types';
|
|
5
|
+
import debug from 'debug';
|
|
6
|
+
import FormData from 'form-data';
|
|
7
|
+
import fs from 'fs-extra';
|
|
8
|
+
import fetch, { RequestInfo, RequestInit, Response } from 'node-fetch';
|
|
9
|
+
|
|
10
|
+
import { PublisherERSConfig } from './Config';
|
|
11
|
+
|
|
12
|
+
const d = debug('electron-forge:publish:ers');
|
|
13
|
+
|
|
14
|
+
interface ERSVersion {
|
|
15
|
+
name: string;
|
|
16
|
+
assets: { name: string }[];
|
|
17
|
+
flavor?: string;
|
|
18
|
+
}
|
|
19
|
+
|
|
20
|
+
const fetchAndCheckStatus = async (url: RequestInfo, init?: RequestInit): Promise<Response> => {
|
|
21
|
+
const result = await fetch(url, init);
|
|
22
|
+
if (result.ok) {
|
|
23
|
+
// res.status >= 200 && res.status < 300
|
|
24
|
+
return result;
|
|
25
|
+
}
|
|
26
|
+
throw new Error(`ERS publish failed with status code: ${result.status} (${result.url})`);
|
|
27
|
+
};
|
|
28
|
+
|
|
29
|
+
export const ersPlatform = (platform: ForgePlatform, arch: ForgeArch): string => {
|
|
30
|
+
switch (platform) {
|
|
31
|
+
case 'darwin':
|
|
32
|
+
return 'osx_64';
|
|
33
|
+
case 'linux':
|
|
34
|
+
return arch === 'ia32' ? 'linux_32' : 'linux_64';
|
|
35
|
+
case 'win32':
|
|
36
|
+
return arch === 'ia32' ? 'windows_32' : 'windows_64';
|
|
37
|
+
default:
|
|
38
|
+
return platform;
|
|
39
|
+
}
|
|
40
|
+
};
|
|
41
|
+
|
|
42
|
+
export default class PublisherERS extends PublisherBase<PublisherERSConfig> {
|
|
43
|
+
name = 'electron-release-server';
|
|
44
|
+
|
|
45
|
+
async publish({ makeResults, setStatusLine }: PublisherOptions): Promise<void> {
|
|
46
|
+
const { config } = this;
|
|
47
|
+
|
|
48
|
+
if (!(config.baseUrl && config.username && config.password)) {
|
|
49
|
+
throw new Error(
|
|
50
|
+
'In order to publish to ERS you must set the "electronReleaseServer.baseUrl", "electronReleaseServer.username" and "electronReleaseServer.password" properties in your Forge config. See the docs for more info'
|
|
51
|
+
);
|
|
52
|
+
}
|
|
53
|
+
|
|
54
|
+
d('attempting to authenticate to ERS');
|
|
55
|
+
|
|
56
|
+
const api = (apiPath: string) => `${config.baseUrl}/${apiPath}`;
|
|
57
|
+
|
|
58
|
+
const { token } = await (
|
|
59
|
+
await fetchAndCheckStatus(api('api/auth/login'), {
|
|
60
|
+
method: 'POST',
|
|
61
|
+
body: JSON.stringify({
|
|
62
|
+
username: config.username,
|
|
63
|
+
password: config.password,
|
|
64
|
+
}),
|
|
65
|
+
headers: {
|
|
66
|
+
'Content-Type': 'application/json',
|
|
67
|
+
},
|
|
68
|
+
})
|
|
69
|
+
).json();
|
|
70
|
+
|
|
71
|
+
const authFetch = (apiPath: string, options?: RequestInit) =>
|
|
72
|
+
fetchAndCheckStatus(api(apiPath), { ...(options || {}), headers: { ...(options || {}).headers, Authorization: `Bearer ${token}` } });
|
|
73
|
+
|
|
74
|
+
const versions: ERSVersion[] = await (await authFetch('api/version')).json();
|
|
75
|
+
const flavor = config.flavor || 'default';
|
|
76
|
+
|
|
77
|
+
for (const makeResult of makeResults) {
|
|
78
|
+
const { packageJSON } = makeResult;
|
|
79
|
+
const artifacts = makeResult.artifacts.filter((artifactPath) => path.basename(artifactPath).toLowerCase() !== 'releases');
|
|
80
|
+
|
|
81
|
+
const existingVersion = versions.find((version) => {
|
|
82
|
+
return version.name === packageJSON.version && (!version.flavor || version.flavor === flavor);
|
|
83
|
+
});
|
|
84
|
+
|
|
85
|
+
let channel = 'stable';
|
|
86
|
+
if (config.channel) {
|
|
87
|
+
channel = config.channel;
|
|
88
|
+
} else if (packageJSON.version.includes('beta')) {
|
|
89
|
+
channel = 'beta';
|
|
90
|
+
} else if (packageJSON.version.includes('alpha')) {
|
|
91
|
+
channel = 'alpha';
|
|
92
|
+
}
|
|
93
|
+
|
|
94
|
+
if (!existingVersion) {
|
|
95
|
+
await authFetch('api/version', {
|
|
96
|
+
method: 'POST',
|
|
97
|
+
body: JSON.stringify({
|
|
98
|
+
channel: {
|
|
99
|
+
name: channel,
|
|
100
|
+
},
|
|
101
|
+
flavor: config.flavor,
|
|
102
|
+
name: packageJSON.version,
|
|
103
|
+
notes: '',
|
|
104
|
+
}),
|
|
105
|
+
headers: {
|
|
106
|
+
'Content-Type': 'application/json',
|
|
107
|
+
},
|
|
108
|
+
});
|
|
109
|
+
}
|
|
110
|
+
|
|
111
|
+
let uploaded = 0;
|
|
112
|
+
const updateStatusLine = () => setStatusLine(`Uploading distributable (${uploaded}/${artifacts.length})`);
|
|
113
|
+
updateStatusLine();
|
|
114
|
+
|
|
115
|
+
await Promise.all(
|
|
116
|
+
artifacts.map(async (artifactPath) => {
|
|
117
|
+
if (existingVersion) {
|
|
118
|
+
const existingAsset = existingVersion.assets.find((asset) => asset.name === path.basename(artifactPath));
|
|
119
|
+
|
|
120
|
+
if (existingAsset) {
|
|
121
|
+
d('asset at path:', artifactPath, 'already exists on server');
|
|
122
|
+
uploaded += 1;
|
|
123
|
+
updateStatusLine();
|
|
124
|
+
return;
|
|
125
|
+
}
|
|
126
|
+
}
|
|
127
|
+
d('attempting to upload asset:', artifactPath);
|
|
128
|
+
const artifactForm = new FormData();
|
|
129
|
+
artifactForm.append('token', token);
|
|
130
|
+
artifactForm.append('version', packageJSON.version);
|
|
131
|
+
artifactForm.append('platform', ersPlatform(makeResult.platform, makeResult.arch));
|
|
132
|
+
|
|
133
|
+
// see https://github.com/form-data/form-data/issues/426
|
|
134
|
+
const fileOptions = {
|
|
135
|
+
knownLength: fs.statSync(artifactPath).size,
|
|
136
|
+
};
|
|
137
|
+
artifactForm.append('file', fs.createReadStream(artifactPath), fileOptions);
|
|
138
|
+
|
|
139
|
+
await authFetch('api/asset', {
|
|
140
|
+
method: 'POST',
|
|
141
|
+
body: artifactForm,
|
|
142
|
+
headers: artifactForm.getHeaders(),
|
|
143
|
+
});
|
|
144
|
+
d('upload successful for asset:', artifactPath);
|
|
145
|
+
uploaded += 1;
|
|
146
|
+
updateStatusLine();
|
|
147
|
+
})
|
|
148
|
+
);
|
|
149
|
+
}
|
|
150
|
+
}
|
|
151
|
+
}
|
|
152
|
+
|
|
153
|
+
export { PublisherERS, PublisherERSConfig };
|
|
@@ -0,0 +1,223 @@
|
|
|
1
|
+
import { ForgeMakeResult, ResolvedForgeConfig } from '@electron-forge/shared-types';
|
|
2
|
+
import { expect } from 'chai';
|
|
3
|
+
import fetchMock from 'fetch-mock';
|
|
4
|
+
import proxyquire from 'proxyquire';
|
|
5
|
+
import { stub } from 'sinon';
|
|
6
|
+
|
|
7
|
+
import type { PublisherERS as PublisherERSType } from '../src/PublisherERS';
|
|
8
|
+
|
|
9
|
+
const noop = () => void 0;
|
|
10
|
+
|
|
11
|
+
describe('PublisherERS', () => {
|
|
12
|
+
let fetch: typeof fetchMock;
|
|
13
|
+
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|
14
|
+
let PublisherERS: typeof PublisherERSType;
|
|
15
|
+
|
|
16
|
+
beforeEach(() => {
|
|
17
|
+
fetch = fetchMock.sandbox();
|
|
18
|
+
PublisherERS = proxyquire.noCallThru().load('../src/PublisherERS', {
|
|
19
|
+
'node-fetch': fetch,
|
|
20
|
+
'fs-extra': { createReadStream: stub().returns(''), statSync: stub().returns({ size: 100 }) },
|
|
21
|
+
}).default;
|
|
22
|
+
});
|
|
23
|
+
|
|
24
|
+
afterEach(() => {
|
|
25
|
+
fetch.restore();
|
|
26
|
+
});
|
|
27
|
+
|
|
28
|
+
describe('new version', () => {
|
|
29
|
+
it('can publish a new version to ERS', async () => {
|
|
30
|
+
const baseUrl = 'https://example.com';
|
|
31
|
+
const token = 'FAKE_TOKEN';
|
|
32
|
+
const flavor = 'lite';
|
|
33
|
+
const version = '3.0.0';
|
|
34
|
+
|
|
35
|
+
// mock login
|
|
36
|
+
fetch.postOnce('path:/api/auth/login', { body: { token }, status: 200 });
|
|
37
|
+
// mock fetch all existing versions
|
|
38
|
+
fetch.getOnce('path:/api/version', { body: [{ name: '2.0.0', assets: [], flavor: 'default' }], status: 200 });
|
|
39
|
+
// mock creating a new version
|
|
40
|
+
fetch.postOnce('path:/api/version', { status: 200 });
|
|
41
|
+
// mock asset upload
|
|
42
|
+
fetch.post('path:/api/asset', { status: 200 });
|
|
43
|
+
|
|
44
|
+
const publisher = new PublisherERS({
|
|
45
|
+
baseUrl,
|
|
46
|
+
username: 'test',
|
|
47
|
+
password: 'test',
|
|
48
|
+
flavor,
|
|
49
|
+
});
|
|
50
|
+
|
|
51
|
+
const makeResults: ForgeMakeResult[] = [
|
|
52
|
+
{
|
|
53
|
+
artifacts: ['/path/to/artifact'],
|
|
54
|
+
packageJSON: {
|
|
55
|
+
version,
|
|
56
|
+
},
|
|
57
|
+
platform: 'linux',
|
|
58
|
+
arch: 'x64',
|
|
59
|
+
},
|
|
60
|
+
];
|
|
61
|
+
|
|
62
|
+
await publisher.publish({ makeResults, dir: '', forgeConfig: {} as ResolvedForgeConfig, setStatusLine: noop });
|
|
63
|
+
|
|
64
|
+
const calls = fetch.calls();
|
|
65
|
+
|
|
66
|
+
// creates a new version with the correct flavor, name, and channel
|
|
67
|
+
expect(calls[2][0]).to.equal(`${baseUrl}/api/version`);
|
|
68
|
+
expect(calls[2][1]?.body).to.equal(`{"channel":{"name":"stable"},"flavor":"${flavor}","name":"${version}","notes":""}`);
|
|
69
|
+
|
|
70
|
+
// uploads asset successfully
|
|
71
|
+
expect(calls[3][0]).to.equal(`${baseUrl}/api/asset`);
|
|
72
|
+
});
|
|
73
|
+
});
|
|
74
|
+
|
|
75
|
+
describe('existing version', () => {
|
|
76
|
+
it('can add new assets', async () => {
|
|
77
|
+
const baseUrl = 'https://example.com';
|
|
78
|
+
const token = 'FAKE_TOKEN';
|
|
79
|
+
const channel = 'stable';
|
|
80
|
+
const flavor = 'lite';
|
|
81
|
+
const version = '2.0.0';
|
|
82
|
+
|
|
83
|
+
// mock login
|
|
84
|
+
fetch.postOnce('path:/api/auth/login', { body: { token }, status: 200 });
|
|
85
|
+
// mock fetch all existing versions
|
|
86
|
+
fetch.getOnce('path:/api/version', { body: [{ name: '2.0.0', assets: [], flavor: 'lite' }], status: 200 });
|
|
87
|
+
// mock asset upload
|
|
88
|
+
fetch.post('path:/api/asset', { status: 200 });
|
|
89
|
+
|
|
90
|
+
const publisher = new PublisherERS({
|
|
91
|
+
baseUrl,
|
|
92
|
+
username: 'test',
|
|
93
|
+
password: 'test',
|
|
94
|
+
channel,
|
|
95
|
+
flavor,
|
|
96
|
+
});
|
|
97
|
+
|
|
98
|
+
const makeResults: ForgeMakeResult[] = [
|
|
99
|
+
{
|
|
100
|
+
artifacts: ['/path/to/artifact'],
|
|
101
|
+
packageJSON: {
|
|
102
|
+
version,
|
|
103
|
+
},
|
|
104
|
+
platform: 'linux',
|
|
105
|
+
arch: 'x64',
|
|
106
|
+
},
|
|
107
|
+
];
|
|
108
|
+
|
|
109
|
+
await publisher.publish({ makeResults, dir: '', forgeConfig: {} as ResolvedForgeConfig, setStatusLine: noop });
|
|
110
|
+
|
|
111
|
+
const calls = fetch.calls();
|
|
112
|
+
|
|
113
|
+
// uploads asset successfully
|
|
114
|
+
expect(calls[2][0]).to.equal(`${baseUrl}/api/asset`);
|
|
115
|
+
});
|
|
116
|
+
|
|
117
|
+
it('does not replace assets for existing version', async () => {
|
|
118
|
+
const baseUrl = 'https://example.com';
|
|
119
|
+
const token = 'FAKE_TOKEN';
|
|
120
|
+
const channel = 'stable';
|
|
121
|
+
const version = '2.0.0';
|
|
122
|
+
|
|
123
|
+
// mock login
|
|
124
|
+
fetch.postOnce('path:/api/auth/login', { body: { token }, status: 200 });
|
|
125
|
+
// mock fetch all existing versions
|
|
126
|
+
fetch.getOnce('path:/api/version', { body: [{ name: '2.0.0', assets: [{ name: 'existing-artifact' }], flavor: 'default' }], status: 200 });
|
|
127
|
+
|
|
128
|
+
const publisher = new PublisherERS({
|
|
129
|
+
baseUrl,
|
|
130
|
+
username: 'test',
|
|
131
|
+
password: 'test',
|
|
132
|
+
channel,
|
|
133
|
+
});
|
|
134
|
+
|
|
135
|
+
const makeResults: ForgeMakeResult[] = [
|
|
136
|
+
{
|
|
137
|
+
artifacts: ['/path/to/existing-artifact'],
|
|
138
|
+
packageJSON: {
|
|
139
|
+
version,
|
|
140
|
+
},
|
|
141
|
+
platform: 'linux',
|
|
142
|
+
arch: 'x64',
|
|
143
|
+
},
|
|
144
|
+
];
|
|
145
|
+
|
|
146
|
+
await publisher.publish({ makeResults, dir: '', forgeConfig: {} as ResolvedForgeConfig, setStatusLine: noop });
|
|
147
|
+
|
|
148
|
+
const calls = fetch.calls();
|
|
149
|
+
expect(calls).to.have.length(2);
|
|
150
|
+
});
|
|
151
|
+
|
|
152
|
+
it('can upload a new flavor for an existing version', async () => {
|
|
153
|
+
const baseUrl = 'https://example.com';
|
|
154
|
+
const token = 'FAKE_TOKEN';
|
|
155
|
+
const version = '2.0.0';
|
|
156
|
+
const flavor = 'lite';
|
|
157
|
+
|
|
158
|
+
// mock login
|
|
159
|
+
fetch.postOnce('path:/api/auth/login', { body: { token }, status: 200 });
|
|
160
|
+
// mock fetch all existing versions
|
|
161
|
+
fetch.getOnce('path:/api/version', { body: [{ name: '2.0.0', assets: [{ name: 'existing-artifact' }], flavor: 'default' }], status: 200 });
|
|
162
|
+
// mock creating a new version
|
|
163
|
+
fetch.postOnce('path:/api/version', { status: 200 });
|
|
164
|
+
// mock asset upload
|
|
165
|
+
fetch.post('path:/api/asset', { status: 200 });
|
|
166
|
+
|
|
167
|
+
const publisher = new PublisherERS({
|
|
168
|
+
baseUrl,
|
|
169
|
+
username: 'test',
|
|
170
|
+
password: 'test',
|
|
171
|
+
flavor,
|
|
172
|
+
});
|
|
173
|
+
|
|
174
|
+
const makeResults: ForgeMakeResult[] = [
|
|
175
|
+
{
|
|
176
|
+
artifacts: ['/path/to/artifact'],
|
|
177
|
+
packageJSON: {
|
|
178
|
+
version,
|
|
179
|
+
},
|
|
180
|
+
platform: 'linux',
|
|
181
|
+
arch: 'x64',
|
|
182
|
+
},
|
|
183
|
+
];
|
|
184
|
+
|
|
185
|
+
await publisher.publish({ makeResults, dir: '', forgeConfig: {} as ResolvedForgeConfig, setStatusLine: noop });
|
|
186
|
+
|
|
187
|
+
const calls = fetch.calls();
|
|
188
|
+
|
|
189
|
+
// creates a new version with the correct flavor, name, and channel
|
|
190
|
+
expect(calls[2][0]).to.equal(`${baseUrl}/api/version`);
|
|
191
|
+
expect(calls[2][1]?.body).to.equal(`{"channel":{"name":"stable"},"flavor":"${flavor}","name":"${version}","notes":""}`);
|
|
192
|
+
|
|
193
|
+
// uploads asset successfully
|
|
194
|
+
expect(calls[3][0]).to.equal(`${baseUrl}/api/asset`);
|
|
195
|
+
});
|
|
196
|
+
|
|
197
|
+
// TODO: implement edge cases
|
|
198
|
+
it('can read the channel from the package.json version');
|
|
199
|
+
it('does not upload the RELEASES file');
|
|
200
|
+
});
|
|
201
|
+
|
|
202
|
+
it('fails if username and password are not provided', () => {
|
|
203
|
+
// @ts-expect-error testing invalid options
|
|
204
|
+
const publisher = new PublisherERS({});
|
|
205
|
+
|
|
206
|
+
expect(publisher.publish({ makeResults: [], dir: '', forgeConfig: {} as ResolvedForgeConfig, setStatusLine: noop })).to.eventually.be.rejectedWith(
|
|
207
|
+
'In order to publish to ERS you must set the "electronReleaseServer.baseUrl", "electronReleaseServer.username" and "electronReleaseServer.password" properties in your Forge config. See the docs for more info'
|
|
208
|
+
);
|
|
209
|
+
});
|
|
210
|
+
|
|
211
|
+
it('fails if the server returns 4xx', async () => {
|
|
212
|
+
fetch.mock('begin:http://example.com', { body: {}, status: 400 });
|
|
213
|
+
|
|
214
|
+
const publisher = new PublisherERS({
|
|
215
|
+
baseUrl: 'http://example.com',
|
|
216
|
+
username: 'test',
|
|
217
|
+
password: 'test',
|
|
218
|
+
});
|
|
219
|
+
return expect(publisher.publish({ makeResults: [], dir: '', forgeConfig: {} as ResolvedForgeConfig, setStatusLine: noop })).to.eventually.be.rejectedWith(
|
|
220
|
+
'ERS publish failed with status code: 400 (http://example.com/api/auth/login)'
|
|
221
|
+
);
|
|
222
|
+
});
|
|
223
|
+
});
|
package/dist/Config.d.ts
DELETED
|
@@ -1,34 +0,0 @@
|
|
|
1
|
-
export interface PublisherERSConfig {
|
|
2
|
-
/**
|
|
3
|
-
* The base URL of your instance of ERS.
|
|
4
|
-
*
|
|
5
|
-
* E.g. https://my-update.server.com
|
|
6
|
-
*/
|
|
7
|
-
baseUrl: string;
|
|
8
|
-
/**
|
|
9
|
-
* The username you use to sign in to ERS
|
|
10
|
-
*/
|
|
11
|
-
username: string;
|
|
12
|
-
/**
|
|
13
|
-
* The password you use to sign in to ERS
|
|
14
|
-
*/
|
|
15
|
-
password: string;
|
|
16
|
-
/**
|
|
17
|
-
* The release channel you want to send artifacts to, normally something like
|
|
18
|
-
* "stable", "beta" or "alpha".
|
|
19
|
-
*
|
|
20
|
-
* If left unspecified we will try to infer the channel from your version
|
|
21
|
-
* field in your package.json.
|
|
22
|
-
*
|
|
23
|
-
* Default: stable
|
|
24
|
-
*/
|
|
25
|
-
channel?: string;
|
|
26
|
-
/**
|
|
27
|
-
* The "flavor" of the binary that you want to release to.
|
|
28
|
-
* This is useful if you want to provide multiple versions
|
|
29
|
-
* of the same application version (e.g. full and lite)
|
|
30
|
-
* to end users.
|
|
31
|
-
*/
|
|
32
|
-
flavor?: string;
|
|
33
|
-
}
|
|
34
|
-
//# sourceMappingURL=Config.d.ts.map
|
package/dist/Config.d.ts.map
DELETED
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"file":"Config.d.ts","sourceRoot":"","sources":["../src/Config.ts"],"names":[],"mappings":"AAAA,MAAM,WAAW,kBAAkB;IACjC;;;;OAIG;IACH,OAAO,EAAE,MAAM,CAAC;IAChB;;OAEG;IACH,QAAQ,EAAE,MAAM,CAAC;IACjB;;OAEG;IACH,QAAQ,EAAE,MAAM,CAAC;IACjB;;;;;;;;OAQG;IACH,OAAO,CAAC,EAAE,MAAM,CAAC;IAEjB;;;;;OAKG;IACH,MAAM,CAAC,EAAE,MAAM,CAAC;CACjB"}
|
package/dist/Config.js
DELETED
package/dist/PublisherERS.d.ts
DELETED
|
@@ -1,10 +0,0 @@
|
|
|
1
|
-
import { PublisherBase, PublisherOptions } from '@electron-forge/publisher-base';
|
|
2
|
-
import { ForgeArch, ForgePlatform } from '@electron-forge/shared-types';
|
|
3
|
-
import { PublisherERSConfig } from './Config';
|
|
4
|
-
export declare const ersPlatform: (platform: ForgePlatform, arch: ForgeArch) => string;
|
|
5
|
-
export default class PublisherERS extends PublisherBase<PublisherERSConfig> {
|
|
6
|
-
name: string;
|
|
7
|
-
publish({ makeResults, setStatusLine }: PublisherOptions): Promise<void>;
|
|
8
|
-
}
|
|
9
|
-
export { PublisherERS, PublisherERSConfig };
|
|
10
|
-
//# sourceMappingURL=PublisherERS.d.ts.map
|
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"file":"PublisherERS.d.ts","sourceRoot":"","sources":["../src/PublisherERS.ts"],"names":[],"mappings":"AAEA,OAAO,EAAE,aAAa,EAAE,gBAAgB,EAAE,MAAM,gCAAgC,CAAC;AACjF,OAAO,EAAE,SAAS,EAAE,aAAa,EAAE,MAAM,8BAA8B,CAAC;AAMxE,OAAO,EAAE,kBAAkB,EAAE,MAAM,UAAU,CAAC;AAmB9C,eAAO,MAAM,WAAW,aAAc,aAAa,QAAQ,SAAS,KAAG,MAWtE,CAAC;AAEF,MAAM,CAAC,OAAO,OAAO,YAAa,SAAQ,aAAa,CAAC,kBAAkB,CAAC;IACzE,IAAI,SAA6B;IAE3B,OAAO,CAAC,EAAE,WAAW,EAAE,aAAa,EAAE,EAAE,gBAAgB,GAAG,OAAO,CAAC,IAAI,CAAC;CA0G/E;AAED,OAAO,EAAE,YAAY,EAAE,kBAAkB,EAAE,CAAC"}
|
package/dist/PublisherERS.js
DELETED
|
@@ -1,150 +0,0 @@
|
|
|
1
|
-
"use strict";
|
|
2
|
-
Object.defineProperty(exports, "__esModule", {
|
|
3
|
-
value: true
|
|
4
|
-
});
|
|
5
|
-
Object.defineProperty(exports, "PublisherERSConfig", {
|
|
6
|
-
enumerable: true,
|
|
7
|
-
get: function() {
|
|
8
|
-
return _config.PublisherERSConfig;
|
|
9
|
-
}
|
|
10
|
-
});
|
|
11
|
-
exports.PublisherERS = exports.default = exports.ersPlatform = void 0;
|
|
12
|
-
var _path = _interopRequireDefault(require("path"));
|
|
13
|
-
var _publisherBase = require("@electron-forge/publisher-base");
|
|
14
|
-
var _debug = _interopRequireDefault(require("debug"));
|
|
15
|
-
var _formData = _interopRequireDefault(require("form-data"));
|
|
16
|
-
var _fsExtra = _interopRequireDefault(require("fs-extra"));
|
|
17
|
-
var _nodeFetch = _interopRequireDefault(require("node-fetch"));
|
|
18
|
-
var _config = require("./Config");
|
|
19
|
-
function _interopRequireDefault(obj) {
|
|
20
|
-
return obj && obj.__esModule ? obj : {
|
|
21
|
-
default: obj
|
|
22
|
-
};
|
|
23
|
-
}
|
|
24
|
-
const d = (0, _debug).default('electron-forge:publish:ers');
|
|
25
|
-
const fetchAndCheckStatus = async (url, init)=>{
|
|
26
|
-
const result = await (0, _nodeFetch).default(url, init);
|
|
27
|
-
if (result.ok) {
|
|
28
|
-
// res.status >= 200 && res.status < 300
|
|
29
|
-
return result;
|
|
30
|
-
}
|
|
31
|
-
throw new Error(`ERS publish failed with status code: ${result.status} (${result.url})`);
|
|
32
|
-
};
|
|
33
|
-
const ersPlatform = (platform, arch)=>{
|
|
34
|
-
switch(platform){
|
|
35
|
-
case 'darwin':
|
|
36
|
-
return 'osx_64';
|
|
37
|
-
case 'linux':
|
|
38
|
-
return arch === 'ia32' ? 'linux_32' : 'linux_64';
|
|
39
|
-
case 'win32':
|
|
40
|
-
return arch === 'ia32' ? 'windows_32' : 'windows_64';
|
|
41
|
-
default:
|
|
42
|
-
return platform;
|
|
43
|
-
}
|
|
44
|
-
};
|
|
45
|
-
exports.ersPlatform = ersPlatform;
|
|
46
|
-
class PublisherERS extends _publisherBase.PublisherBase {
|
|
47
|
-
async publish({ makeResults , setStatusLine }) {
|
|
48
|
-
const { config } = this;
|
|
49
|
-
if (!(config.baseUrl && config.username && config.password)) {
|
|
50
|
-
throw new Error('In order to publish to ERS you must set the "electronReleaseServer.baseUrl", "electronReleaseServer.username" and "electronReleaseServer.password" properties in your Forge config. See the docs for more info');
|
|
51
|
-
}
|
|
52
|
-
d('attempting to authenticate to ERS');
|
|
53
|
-
const api = (apiPath)=>`${config.baseUrl}/${apiPath}`
|
|
54
|
-
;
|
|
55
|
-
const { token } = await (await fetchAndCheckStatus(api('api/auth/login'), {
|
|
56
|
-
method: 'POST',
|
|
57
|
-
body: JSON.stringify({
|
|
58
|
-
username: config.username,
|
|
59
|
-
password: config.password
|
|
60
|
-
}),
|
|
61
|
-
headers: {
|
|
62
|
-
'Content-Type': 'application/json'
|
|
63
|
-
}
|
|
64
|
-
})).json();
|
|
65
|
-
const authFetch = (apiPath, options)=>fetchAndCheckStatus(api(apiPath), {
|
|
66
|
-
...options || {},
|
|
67
|
-
headers: {
|
|
68
|
-
...(options || {}).headers,
|
|
69
|
-
Authorization: `Bearer ${token}`
|
|
70
|
-
}
|
|
71
|
-
})
|
|
72
|
-
;
|
|
73
|
-
const versions = await (await authFetch('api/version')).json();
|
|
74
|
-
const flavor = config.flavor || 'default';
|
|
75
|
-
for (const makeResult of makeResults){
|
|
76
|
-
const { packageJSON } = makeResult;
|
|
77
|
-
const artifacts = makeResult.artifacts.filter((artifactPath)=>_path.default.basename(artifactPath).toLowerCase() !== 'releases'
|
|
78
|
-
);
|
|
79
|
-
const existingVersion = versions.find((version)=>{
|
|
80
|
-
return version.name === packageJSON.version && (!version.flavor || version.flavor === flavor);
|
|
81
|
-
});
|
|
82
|
-
let channel = 'stable';
|
|
83
|
-
if (config.channel) {
|
|
84
|
-
channel = config.channel;
|
|
85
|
-
} else if (packageJSON.version.includes('beta')) {
|
|
86
|
-
channel = 'beta';
|
|
87
|
-
} else if (packageJSON.version.includes('alpha')) {
|
|
88
|
-
channel = 'alpha';
|
|
89
|
-
}
|
|
90
|
-
if (!existingVersion) {
|
|
91
|
-
await authFetch('api/version', {
|
|
92
|
-
method: 'POST',
|
|
93
|
-
body: JSON.stringify({
|
|
94
|
-
channel: {
|
|
95
|
-
name: channel
|
|
96
|
-
},
|
|
97
|
-
flavor: config.flavor,
|
|
98
|
-
name: packageJSON.version,
|
|
99
|
-
notes: ''
|
|
100
|
-
}),
|
|
101
|
-
headers: {
|
|
102
|
-
'Content-Type': 'application/json'
|
|
103
|
-
}
|
|
104
|
-
});
|
|
105
|
-
}
|
|
106
|
-
let uploaded = 0;
|
|
107
|
-
const updateStatusLine = ()=>setStatusLine(`Uploading distributable (${uploaded}/${artifacts.length})`)
|
|
108
|
-
;
|
|
109
|
-
updateStatusLine();
|
|
110
|
-
await Promise.all(artifacts.map(async (artifactPath)=>{
|
|
111
|
-
if (existingVersion) {
|
|
112
|
-
const existingAsset = existingVersion.assets.find((asset)=>asset.name === _path.default.basename(artifactPath)
|
|
113
|
-
);
|
|
114
|
-
if (existingAsset) {
|
|
115
|
-
d('asset at path:', artifactPath, 'already exists on server');
|
|
116
|
-
uploaded += 1;
|
|
117
|
-
updateStatusLine();
|
|
118
|
-
return;
|
|
119
|
-
}
|
|
120
|
-
}
|
|
121
|
-
d('attempting to upload asset:', artifactPath);
|
|
122
|
-
const artifactForm = new _formData.default();
|
|
123
|
-
artifactForm.append('token', token);
|
|
124
|
-
artifactForm.append('version', packageJSON.version);
|
|
125
|
-
artifactForm.append('platform', ersPlatform(makeResult.platform, makeResult.arch));
|
|
126
|
-
// see https://github.com/form-data/form-data/issues/426
|
|
127
|
-
const fileOptions = {
|
|
128
|
-
knownLength: _fsExtra.default.statSync(artifactPath).size
|
|
129
|
-
};
|
|
130
|
-
artifactForm.append('file', _fsExtra.default.createReadStream(artifactPath), fileOptions);
|
|
131
|
-
await authFetch('api/asset', {
|
|
132
|
-
method: 'POST',
|
|
133
|
-
body: artifactForm,
|
|
134
|
-
headers: artifactForm.getHeaders()
|
|
135
|
-
});
|
|
136
|
-
d('upload successful for asset:', artifactPath);
|
|
137
|
-
uploaded += 1;
|
|
138
|
-
updateStatusLine();
|
|
139
|
-
}));
|
|
140
|
-
}
|
|
141
|
-
}
|
|
142
|
-
constructor(...args){
|
|
143
|
-
super(...args);
|
|
144
|
-
this.name = 'electron-release-server';
|
|
145
|
-
}
|
|
146
|
-
}
|
|
147
|
-
exports.default = PublisherERS;
|
|
148
|
-
exports.PublisherERS = PublisherERS;
|
|
149
|
-
|
|
150
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../src/PublisherERS.ts"],"sourcesContent":["import path from 'path';\n\nimport { PublisherBase, PublisherOptions } from '@electron-forge/publisher-base';\nimport { ForgeArch, ForgePlatform } from '@electron-forge/shared-types';\nimport debug from 'debug';\nimport FormData from 'form-data';\nimport fs from 'fs-extra';\nimport fetch, { RequestInfo, RequestInit, Response } from 'node-fetch';\n\nimport { PublisherERSConfig } from './Config';\n\nconst d = debug('electron-forge:publish:ers');\n\ninterface ERSVersion {\n  name: string;\n  assets: { name: string }[];\n  flavor?: string;\n}\n\nconst fetchAndCheckStatus = async (url: RequestInfo, init?: RequestInit): Promise<Response> => {\n  const result = await fetch(url, init);\n  if (result.ok) {\n    // res.status >= 200 && res.status < 300\n    return result;\n  }\n  throw new Error(`ERS publish failed with status code: ${result.status} (${result.url})`);\n};\n\nexport const ersPlatform = (platform: ForgePlatform, arch: ForgeArch): string => {\n  switch (platform) {\n    case 'darwin':\n      return 'osx_64';\n    case 'linux':\n      return arch === 'ia32' ? 'linux_32' : 'linux_64';\n    case 'win32':\n      return arch === 'ia32' ? 'windows_32' : 'windows_64';\n    default:\n      return platform;\n  }\n};\n\nexport default class PublisherERS extends PublisherBase<PublisherERSConfig> {\n  name = 'electron-release-server';\n\n  async publish({ makeResults, setStatusLine }: PublisherOptions): Promise<void> {\n    const { config } = this;\n\n    if (!(config.baseUrl && config.username && config.password)) {\n      throw new Error(\n        'In order to publish to ERS you must set the \"electronReleaseServer.baseUrl\", \"electronReleaseServer.username\" and \"electronReleaseServer.password\" properties in your Forge config. See the docs for more info'\n      );\n    }\n\n    d('attempting to authenticate to ERS');\n\n    const api = (apiPath: string) => `${config.baseUrl}/${apiPath}`;\n\n    const { token } = await (\n      await fetchAndCheckStatus(api('api/auth/login'), {\n        method: 'POST',\n        body: JSON.stringify({\n          username: config.username,\n          password: config.password,\n        }),\n        headers: {\n          'Content-Type': 'application/json',\n        },\n      })\n    ).json();\n\n    const authFetch = (apiPath: string, options?: RequestInit) =>\n      fetchAndCheckStatus(api(apiPath), { ...(options || {}), headers: { ...(options || {}).headers, Authorization: `Bearer ${token}` } });\n\n    const versions: ERSVersion[] = await (await authFetch('api/version')).json();\n    const flavor = config.flavor || 'default';\n\n    for (const makeResult of makeResults) {\n      const { packageJSON } = makeResult;\n      const artifacts = makeResult.artifacts.filter((artifactPath) => path.basename(artifactPath).toLowerCase() !== 'releases');\n\n      const existingVersion = versions.find((version) => {\n        return version.name === packageJSON.version && (!version.flavor || version.flavor === flavor);\n      });\n\n      let channel = 'stable';\n      if (config.channel) {\n        channel = config.channel;\n      } else if (packageJSON.version.includes('beta')) {\n        channel = 'beta';\n      } else if (packageJSON.version.includes('alpha')) {\n        channel = 'alpha';\n      }\n\n      if (!existingVersion) {\n        await authFetch('api/version', {\n          method: 'POST',\n          body: JSON.stringify({\n            channel: {\n              name: channel,\n            },\n            flavor: config.flavor,\n            name: packageJSON.version,\n            notes: '',\n          }),\n          headers: {\n            'Content-Type': 'application/json',\n          },\n        });\n      }\n\n      let uploaded = 0;\n      const updateStatusLine = () => setStatusLine(`Uploading distributable (${uploaded}/${artifacts.length})`);\n      updateStatusLine();\n\n      await Promise.all(\n        artifacts.map(async (artifactPath) => {\n          if (existingVersion) {\n            const existingAsset = existingVersion.assets.find((asset) => asset.name === path.basename(artifactPath));\n\n            if (existingAsset) {\n              d('asset at path:', artifactPath, 'already exists on server');\n              uploaded += 1;\n              updateStatusLine();\n              return;\n            }\n          }\n          d('attempting to upload asset:', artifactPath);\n          const artifactForm = new FormData();\n          artifactForm.append('token', token);\n          artifactForm.append('version', packageJSON.version);\n          artifactForm.append('platform', ersPlatform(makeResult.platform, makeResult.arch));\n\n          // see https://github.com/form-data/form-data/issues/426\n          const fileOptions = {\n            knownLength: fs.statSync(artifactPath).size,\n          };\n          artifactForm.append('file', fs.createReadStream(artifactPath), fileOptions);\n\n          await authFetch('api/asset', {\n            method: 'POST',\n            body: artifactForm,\n            headers: artifactForm.getHeaders(),\n          });\n          d('upload successful for asset:', artifactPath);\n          uploaded += 1;\n          updateStatusLine();\n        })\n      );\n    }\n  }\n}\n\nexport { PublisherERS, PublisherERSConfig };\n"],"names":["PublisherERSConfig","d","debug","fetchAndCheckStatus","url","init","result","fetch","ok","Error","status","ersPlatform","platform","arch","PublisherERS","PublisherBase","publish","makeResults","setStatusLine","config","baseUrl","username","password","api","apiPath","token","method","body","JSON","stringify","headers","json","authFetch","options","Authorization","versions","flavor","makeResult","packageJSON","artifacts","filter","artifactPath","path","basename","toLowerCase","existingVersion","find","version","name","channel","includes","notes","uploaded","updateStatusLine","length","Promise","all","map","existingAsset","assets","asset","artifactForm","FormData","append","fileOptions","knownLength","fs","statSync","size","createReadStream","getHeaders"],"mappings":"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"}
|