restServer.ts 3.26 KB
Newer Older
liuzhe-lz's avatar
liuzhe-lz committed
1
2
// Copyright (c) Microsoft Corporation.
// Licensed under the MIT license.
Deshui Yu's avatar
Deshui Yu committed
3

4
5
6
import assert from 'assert';
import express from 'express';
import http from 'http';
Deshui Yu's avatar
Deshui Yu committed
7
import { Deferred } from 'ts-deferred';
8
import { getLogger, Logger } from './log';
9
import { getBasePort } from './experimentStartupInfo';
Deshui Yu's avatar
Deshui Yu committed
10

SparkSnail's avatar
SparkSnail committed
11

12
13
/**
 * Abstraction class to create a RestServer
14
 * The module who wants to use a RestServer could <b>extends</b> this abstract class
15
16
17
 * And implement its own registerRestHandler() function to register routers
 */
export abstract class RestServer {
Deshui Yu's avatar
Deshui Yu committed
18
19
20
21
    private startTask!: Deferred<void>;
    private stopTask!: Deferred<void>;
    private server!: http.Server;

22
23
24
25
    /** The fields can be inherited by subclass */
    protected hostName: string = '0.0.0.0';
    protected port?: number;
    protected app: express.Application = express();
liuzhe-lz's avatar
liuzhe-lz committed
26
    protected log: Logger = getLogger('RestServer');
27
    protected basePort?: number;
28

29
30
31
32
33
    constructor() {
        this.port = getBasePort();
        assert(this.port && this.port > 1024);
    }

Deshui Yu's avatar
Deshui Yu committed
34
35
36
37
    get endPoint(): string {
        return `http://${this.hostName}:${this.port}`;
    }

38
39
    public start(hostName?: string): Promise<void> {
        this.log.info(`RestServer start`);
Deshui Yu's avatar
Deshui Yu committed
40
41
42
43
44
45
46
47
48
49
        if (this.startTask !== undefined) {
            return this.startTask.promise;
        }
        this.startTask = new Deferred<void>();

        this.registerRestHandler();

        if (hostName) {
            this.hostName = hostName;
        }
50
51

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

53
        this.server = this.app.listen(this.port as number, this.hostName).on('listening', () => {
Deshui Yu's avatar
Deshui Yu committed
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
            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
75
76
77
                    //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.
SparkSnail's avatar
SparkSnail committed
78
                    //Refer https://nodejs.org/docs/latest/api/net.html#net_server_close_callback
Deshui Yu's avatar
Deshui Yu committed
79
80
81
82
83
84
85
86
87
88
89
90
91
                    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
92
        this.stopTask.resolve()
Deshui Yu's avatar
Deshui Yu committed
93
94
95
        return this.stopTask.promise;
    }

96
97
98
99
    /**
     * Register REST handler, which is left for subclass to implement
     */
    protected abstract registerRestHandler(): void;
Deshui Yu's avatar
Deshui Yu committed
100
}