Book a Demo!
CoCalc Logo Icon
StoreFeaturesDocsShareSupportNewsAboutPoliciesSign UpSign In
microsoft
GitHub Repository: microsoft/vscode
Path: blob/main/src/vs/workbench/api/node/loopbackServer.ts
3296 views
1
/*---------------------------------------------------------------------------------------------
2
* Copyright (c) Microsoft Corporation. All rights reserved.
3
* Licensed under the MIT License. See License.txt in the project root for license information.
4
*--------------------------------------------------------------------------------------------*/
5
import { randomBytes } from 'crypto';
6
import * as http from 'http';
7
import { URL } from 'url';
8
import { DeferredPromise } from '../../../base/common/async.js';
9
import { DEFAULT_AUTH_FLOW_PORT } from '../../../base/common/oauth.js';
10
import { URI } from '../../../base/common/uri.js';
11
import { ILogger } from '../../../platform/log/common/log.js';
12
13
export interface IOAuthResult {
14
code: string;
15
state: string;
16
}
17
18
export interface ILoopbackServer {
19
/**
20
* The state parameter used in the OAuth flow.
21
*/
22
readonly state: string;
23
24
/**
25
* Starts the server.
26
* @throws If the server fails to start.
27
* @throws If the server is already started.
28
*/
29
start(): Promise<void>;
30
31
/**
32
* Stops the server.
33
* @throws If the server is not started.
34
* @throws If the server fails to stop.
35
*/
36
stop(): Promise<void>;
37
38
/**
39
* Returns a promise that resolves to the result of the OAuth flow.
40
*/
41
waitForOAuthResponse(): Promise<IOAuthResult>;
42
}
43
44
export class LoopbackAuthServer implements ILoopbackServer {
45
private readonly _server: http.Server;
46
private readonly _resultPromise: Promise<IOAuthResult>;
47
48
private _state = randomBytes(16).toString('base64');
49
private _port: number | undefined;
50
51
constructor(
52
private readonly _logger: ILogger,
53
private readonly _appUri: URI,
54
private readonly _appName: string
55
) {
56
const deferredPromise = new DeferredPromise<IOAuthResult>();
57
this._resultPromise = deferredPromise.p;
58
59
this._server = http.createServer((req, res) => {
60
const reqUrl = new URL(req.url!, `http://${req.headers.host}`);
61
switch (reqUrl.pathname) {
62
case '/': {
63
const code = reqUrl.searchParams.get('code') ?? undefined;
64
const state = reqUrl.searchParams.get('state') ?? undefined;
65
const error = reqUrl.searchParams.get('error') ?? undefined;
66
if (error) {
67
res.writeHead(302, { location: `/done?error=${reqUrl.searchParams.get('error_description') || error}` });
68
res.end();
69
deferredPromise.error(new Error(error));
70
break;
71
}
72
if (!code || !state) {
73
res.writeHead(400);
74
res.end();
75
break;
76
}
77
if (this.state !== state) {
78
res.writeHead(302, { location: `/done?error=${encodeURIComponent('State does not match.')}` });
79
res.end();
80
deferredPromise.error(new Error('State does not match.'));
81
break;
82
}
83
deferredPromise.complete({ code, state });
84
res.writeHead(302, { location: '/done' });
85
res.end();
86
break;
87
}
88
// Serve the static files
89
case '/done':
90
this._sendPage(res);
91
break;
92
default:
93
res.writeHead(404);
94
res.end();
95
break;
96
}
97
});
98
}
99
100
get state(): string { return this._state; }
101
get redirectUri(): string {
102
if (this._port === undefined) {
103
throw new Error('Server is not started yet');
104
}
105
return `http://127.0.0.1:${this._port}/`;
106
}
107
108
private _sendPage(res: http.ServerResponse): void {
109
const html = this.getHtml();
110
res.writeHead(200, {
111
'Content-Type': 'text/html',
112
'Content-Length': Buffer.byteLength(html, 'utf8')
113
});
114
res.end(html);
115
}
116
117
start(): Promise<void> {
118
const deferredPromise = new DeferredPromise<void>();
119
if (this._server.listening) {
120
throw new Error('Server is already started');
121
}
122
const portTimeout = setTimeout(() => {
123
deferredPromise.error(new Error('Timeout waiting for port'));
124
}, 5000);
125
this._server.on('listening', () => {
126
const address = this._server.address();
127
if (typeof address === 'string') {
128
this._port = parseInt(address);
129
} else if (address instanceof Object) {
130
this._port = address.port;
131
} else {
132
throw new Error('Unable to determine port');
133
}
134
135
clearTimeout(portTimeout);
136
deferredPromise.complete();
137
});
138
this._server.on('error', err => {
139
if ('code' in err && err.code === 'EADDRINUSE') {
140
this._logger.error('Address in use, retrying with a different port...');
141
// Best effort to use a specific port, but fallback to a random one if it is in use
142
this._server.listen(0, '127.0.0.1');
143
return;
144
}
145
clearTimeout(portTimeout);
146
deferredPromise.error(new Error(`Error listening to server: ${err}`));
147
});
148
this._server.on('close', () => {
149
deferredPromise.error(new Error('Closed'));
150
});
151
// Best effort to use a specific port, but fallback to a random one if it is in use
152
this._server.listen(DEFAULT_AUTH_FLOW_PORT, '127.0.0.1');
153
return deferredPromise.p;
154
}
155
156
stop(): Promise<void> {
157
const deferredPromise = new DeferredPromise<void>();
158
if (!this._server.listening) {
159
deferredPromise.complete();
160
return deferredPromise.p;
161
}
162
this._server.close((err) => {
163
if (err) {
164
deferredPromise.error(err);
165
} else {
166
deferredPromise.complete();
167
}
168
});
169
// If the server is not closed within 5 seconds, reject the promise
170
setTimeout(() => {
171
if (!deferredPromise.isResolved) {
172
deferredPromise.error(new Error('Timeout waiting for server to close'));
173
}
174
}, 5000);
175
return deferredPromise.p;
176
}
177
178
waitForOAuthResponse(): Promise<IOAuthResult> {
179
return this._resultPromise;
180
}
181
182
getHtml(): string {
183
// TODO: Bring this in via mixin. Skipping exploration for now.
184
let backgroundImage = 'url(\'\')';
185
if (this._appName === 'Visual Studio Code') {
186
backgroundImage = 'url(\'\')';
187
} else if (this._appName === 'Visual Studio Code - Insiders') {
188
backgroundImage = 'url(\'\')';
189
}
190
return `<!DOCTYPE html>
191
<html lang="en">
192
193
<head>
194
<meta charset="utf-8" />
195
<title>GitHub Authentication - Sign In</title>
196
<meta name="viewport" content="width=device-width, initial-scale=1">
197
<style>
198
html {
199
height: 100%;
200
}
201
202
body {
203
box-sizing: border-box;
204
min-height: 100%;
205
margin: 0;
206
padding: 15px 30px;
207
display: flex;
208
flex-direction: column;
209
color: white;
210
font-family: "Segoe UI","Helvetica Neue","Helvetica",Arial,sans-serif;
211
background-color: #2C2C32;
212
}
213
214
.branding {
215
background-image: ${backgroundImage};
216
background-size: 24px;
217
background-repeat: no-repeat;
218
background-position: left center;
219
padding-left: 36px;
220
font-size: 20px;
221
letter-spacing: -0.04rem;
222
font-weight: 400;
223
color: white;
224
text-decoration: none;
225
}
226
227
.message-container {
228
flex-grow: 1;
229
display: flex;
230
align-items: center;
231
justify-content: center;
232
margin: 0 30px;
233
}
234
235
.message {
236
font-weight: 300;
237
font-size: 1.4rem;
238
}
239
240
body.error .message {
241
display: none;
242
}
243
244
body.error .error-message {
245
display: block;
246
}
247
248
.error-message {
249
display: none;
250
font-weight: 300;
251
font-size: 1.3rem;
252
}
253
254
.error-text {
255
color: red;
256
font-size: 1rem;
257
}
258
259
@font-face {
260
font-family: 'Segoe UI';
261
src: url("https://c.s-microsoft.com/static/fonts/segoe-ui/west-european/light/latest.eot"),url("https://c.s-microsoft.com/static/fonts/segoe-ui/west-european/light/latest.eot?#iefix") format("embedded-opentype");
262
src: local("Segoe UI Light"),url("https://c.s-microsoft.com/static/fonts/segoe-ui/west-european/light/latest.woff2") format("woff2"),url("https://c.s-microsoft.com/static/fonts/segoe-ui/west-european/light/latest.woff") format("woff"),url("https://c.s-microsoft.com/static/fonts/segoe-ui/west-european/light/latest.ttf") format("truetype"),url("https://c.s-microsoft.com/static/fonts/segoe-ui/west-european/light/latest.svg#web") format("svg");
263
font-weight: 200
264
}
265
266
@font-face {
267
font-family: 'Segoe UI';
268
src: url("https://c.s-microsoft.com/static/fonts/segoe-ui/west-european/semilight/latest.eot"),url("https://c.s-microsoft.com/static/fonts/segoe-ui/west-european/semilight/latest.eot?#iefix") format("embedded-opentype");
269
src: local("Segoe UI Semilight"),url("https://c.s-microsoft.com/static/fonts/segoe-ui/west-european/semilight/latest.woff2") format("woff2"),url("https://c.s-microsoft.com/static/fonts/segoe-ui/west-european/semilight/latest.woff") format("woff"),url("https://c.s-microsoft.com/static/fonts/segoe-ui/west-european/semilight/latest.ttf") format("truetype"),url("https://c.s-microsoft.com/static/fonts/segoe-ui/west-european/semilight/latest.svg#web") format("svg");
270
font-weight: 300
271
}
272
273
@font-face {
274
font-family: 'Segoe UI';
275
src: url("https://c.s-microsoft.com/static/fonts/segoe-ui/west-european/normal/latest.eot"),url("https://c.s-microsoft.com/static/fonts/segoe-ui/west-european/normal/latest.eot?#iefix") format("embedded-opentype");
276
src: local("Segoe UI"),url("https://c.s-microsoft.com/static/fonts/segoe-ui/west-european/normal/latest.woff2") format("woff"),url("https://c.s-microsoft.com/static/fonts/segoe-ui/west-european/normal/latest.woff") format("woff"),url("https://c.s-microsoft.com/static/fonts/segoe-ui/west-european/normal/latest.ttf") format("truetype"),url("https://c.s-microsoft.com/static/fonts/segoe-ui/west-european/normal/latest.svg#web") format("svg");
277
font-weight: 400
278
}
279
280
@font-face {
281
font-family: 'Segoe UI';
282
src: url("https://c.s-microsoft.com/static/fonts/segoe-ui/west-european/semibold/latest.eot"),url("https://c.s-microsoft.com/static/fonts/segoe-ui/west-european/semibold/latest.eot?#iefix") format("embedded-opentype");
283
src: local("Segoe UI Semibold"),url("https://c.s-microsoft.com/static/fonts/segoe-ui/west-european/semibold/latest.woff2") format("woff"),url("https://c.s-microsoft.com/static/fonts/segoe-ui/west-european/semibold/latest.woff") format("woff"),url("https://c.s-microsoft.com/static/fonts/segoe-ui/west-european/semibold/latest.ttf") format("truetype"),url("https://c.s-microsoft.com/static/fonts/segoe-ui/west-european/semibold/latest.svg#web") format("svg");
284
font-weight: 600
285
}
286
287
@font-face {
288
font-family: 'Segoe UI';
289
src: url("https://c.s-microsoft.com/static/fonts/segoe-ui/west-european/bold/latest.eot"),url("https://c.s-microsoft.com/static/fonts/segoe-ui/west-european/bold/latest.eot?#iefix") format("embedded-opentype");
290
src: local("Segoe UI Bold"),url("https://c.s-microsoft.com/static/fonts/segoe-ui/west-european/bold/latest.woff2") format("woff"),url("https://c.s-microsoft.com/static/fonts/segoe-ui/west-european/bold/latest.woff") format("woff"),url("https://c.s-microsoft.com/static/fonts/segoe-ui/west-european/bold/latest.ttf") format("truetype"),url("https://c.s-microsoft.com/static/fonts/segoe-ui/west-european/bold/latest.svg#web") format("svg");
291
font-weight: 700
292
}
293
</style>
294
</head>
295
296
<body>
297
<a class="branding" href="https://code.visualstudio.com/">
298
${this._appName}
299
</a>
300
<div class="message-container">
301
<div class="message">
302
Sign-in successful! Returning to ${this._appName}...
303
<br><br>
304
If you're not redirected automatically, <a href="${this._appUri.toString(true)}" style="color: #85CEFF;">click here</a> or close this page.
305
</div>
306
<div class="error-message">
307
An error occurred while signing in:
308
<div class="error-text"></div>
309
</div>
310
</div>
311
<script>
312
const search = window.location.search;
313
const error = (/[?&^]error=([^&]+)/.exec(search) || [])[1];
314
if (error) {
315
document.querySelector('.error-text')
316
.textContent = decodeURIComponent(error);
317
document.querySelector('body')
318
.classList.add('error');
319
} else {
320
// Redirect to the app URI after a 1-second delay to allow page to load
321
setTimeout(function() {
322
window.location.href = '${this._appUri.toString(true)}';
323
}, 1000);
324
}
325
</script>
326
</body>
327
</html>`;
328
}
329
}
330
331