restServer.ts 4.38 KB
Newer Older
Deshui Yu's avatar
Deshui Yu committed
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
/**
 * Copyright (c) Microsoft Corporation
 * All rights reserved.
 *
 * MIT License
 *
 * Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated
 * documentation files (the "Software"), to deal in the Software without restriction, including without limitation
 * the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and
 * to permit persons to whom the Software is furnished to do so, subject to the following conditions:
 * The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software.
 *
 * THE SOFTWARE IS PROVIDED *AS IS*, WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING
 * BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
 * NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM,
 * DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
 * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
 */

'use strict';

22
import * as assert from 'assert';
Deshui Yu's avatar
Deshui Yu committed
23
24
25
import * as express from 'express';
import * as http from 'http';
import { Deferred } from 'ts-deferred';
26
import { getLogger, Logger } from './log';
27
import { getBasePort } from './experimentStartupInfo';
Deshui Yu's avatar
Deshui Yu committed
28

SparkSnail's avatar
SparkSnail committed
29

30
31
32
33
34
35
/**
 * Abstraction class to create a RestServer
 * The module who wants to use a RestServer could <b>extends</b> this abstract class 
 * And implement its own registerRestHandler() function to register routers
 */
export abstract class RestServer {
Deshui Yu's avatar
Deshui Yu committed
36
37
38
39
    private startTask!: Deferred<void>;
    private stopTask!: Deferred<void>;
    private server!: http.Server;

40
41
42
43
44
    /** The fields can be inherited by subclass */
    protected hostName: string = '0.0.0.0';
    protected port?: number;
    protected app: express.Application = express();
    protected log: Logger = getLogger();
45
    protected basePort?: number;
46
    
47
48
49
50
51
    constructor() {
        this.port = getBasePort();
        assert(this.port && this.port > 1024);
    }

Deshui Yu's avatar
Deshui Yu committed
52
53
54
55
56
    get endPoint(): string {
        // tslint:disable-next-line:no-http-string
        return `http://${this.hostName}:${this.port}`;
    }

57
58
    public start(hostName?: string): Promise<void> {
        this.log.info(`RestServer start`);
Deshui Yu's avatar
Deshui Yu committed
59
60
61
62
63
64
65
66
67
68
        if (this.startTask !== undefined) {
            return this.startTask.promise;
        }
        this.startTask = new Deferred<void>();

        this.registerRestHandler();

        if (hostName) {
            this.hostName = hostName;
        }
69
70

        this.log.info(`RestServer base port is ${this.port}`);
Deshui Yu's avatar
Deshui Yu committed
71

72
        this.server = this.app.listen(this.port as number, this.hostName).on('listening', () => {
Deshui Yu's avatar
Deshui Yu committed
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
            this.startTask.resolve();
        }).on('error', (e: Error) => {
            this.startTask.reject(e);
        });

        return this.startTask.promise;
    }

    public stop(): Promise<void> {
        if (this.stopTask !== undefined) {
            return this.stopTask.promise;
        }
        this.stopTask = new Deferred<void>();

        if (this.startTask === undefined) {
            this.stopTask.resolve();

            return this.stopTask.promise;
        } else {
            this.startTask.promise.then(
                () => { // Started
SparkSnail's avatar
SparkSnail committed
94
95
96
97
                    //Stops the server from accepting new connections and keeps existing connections. 
                    //This function is asynchronous, the server is finally closed when all connections 
                    //are ended and the server emits a 'close' event. 
                    //Refer https://nodejs.org/docs/latest/api/net.html#net_server_close_callback
Deshui Yu's avatar
Deshui Yu committed
98
99
100
101
102
103
104
105
106
107
108
109
110
                    this.server.close().on('close', () => {
                        this.log.info('Rest server stopped.');
                        this.stopTask.resolve();
                    }).on('error', (e: Error) => {
                        this.log.error(`Error occurred stopping Rest server: ${e.message}`);
                        this.stopTask.reject();
                    });
                },
                () => { // Start task rejected
                    this.stopTask.resolve();
                }
            );
        }
SparkSnail's avatar
SparkSnail committed
111
        this.stopTask.resolve()
Deshui Yu's avatar
Deshui Yu committed
112
113
114
        return this.stopTask.promise;
    }

115
116
117
118
    /**
     * Register REST handler, which is left for subclass to implement
     */
    protected abstract registerRestHandler(): void;
Deshui Yu's avatar
Deshui Yu committed
119
}