Built motion from commit 6a09e18b.|2.6.11
[motion2.git] / legacy-libs / grpc-cloned / node_modules / @grpc / grpc-js / build / src / channel.js
diff --git a/legacy-libs/grpc-cloned/node_modules/@grpc/grpc-js/build/src/channel.js b/legacy-libs/grpc-cloned/node_modules/@grpc/grpc-js/build/src/channel.js
new file mode 100644 (file)
index 0000000..2ecca4e
--- /dev/null
@@ -0,0 +1,297 @@
+"use strict";
+Object.defineProperty(exports, "__esModule", { value: true });
+const events_1 = require("events");
+const http2 = require("http2");
+const tls_1 = require("tls");
+const url = require("url");
+const call_credentials_filter_1 = require("./call-credentials-filter");
+const call_stream_1 = require("./call-stream");
+const channel_options_1 = require("./channel-options");
+const compression_filter_1 = require("./compression-filter");
+const constants_1 = require("./constants");
+const deadline_filter_1 = require("./deadline-filter");
+const filter_stack_1 = require("./filter-stack");
+const metadata_status_filter_1 = require("./metadata-status-filter");
+const subchannel_1 = require("./subchannel");
+const { version: clientVersion } = require('../../package.json');
+const MIN_CONNECT_TIMEOUT_MS = 20000;
+const INITIAL_BACKOFF_MS = 1000;
+const BACKOFF_MULTIPLIER = 1.6;
+const MAX_BACKOFF_MS = 120000;
+const BACKOFF_JITTER = 0.2;
+const { HTTP2_HEADER_AUTHORITY, HTTP2_HEADER_CONTENT_TYPE, HTTP2_HEADER_METHOD, HTTP2_HEADER_PATH, HTTP2_HEADER_TE, HTTP2_HEADER_USER_AGENT } = http2.constants;
+var ConnectivityState;
+(function (ConnectivityState) {
+    ConnectivityState[ConnectivityState["CONNECTING"] = 0] = "CONNECTING";
+    ConnectivityState[ConnectivityState["READY"] = 1] = "READY";
+    ConnectivityState[ConnectivityState["TRANSIENT_FAILURE"] = 2] = "TRANSIENT_FAILURE";
+    ConnectivityState[ConnectivityState["IDLE"] = 3] = "IDLE";
+    ConnectivityState[ConnectivityState["SHUTDOWN"] = 4] = "SHUTDOWN";
+})(ConnectivityState = exports.ConnectivityState || (exports.ConnectivityState = {}));
+function uniformRandom(min, max) {
+    return Math.random() * (max - min) + min;
+}
+class Http2Channel extends events_1.EventEmitter {
+    constructor(address, credentials, options) {
+        super();
+        this.credentials = credentials;
+        this.options = options;
+        this.connectivityState = ConnectivityState.IDLE;
+        // Helper Promise object only used in the implementation of connect().
+        this.connecting = null;
+        /* For now, we have up to one subchannel, which will exist as long as we are
+         * connecting or trying to connect */
+        this.subChannel = null;
+        this.subChannelConnectCallback = () => { };
+        this.subChannelCloseCallback = () => { };
+        this.currentBackoff = INITIAL_BACKOFF_MS;
+        for (const option in options) {
+            if (options.hasOwnProperty(option)) {
+                if (!channel_options_1.recognizedOptions.hasOwnProperty(option)) {
+                    console.warn(`Unrecognized channel argument '${option}' will be ignored.`);
+                }
+            }
+        }
+        if (credentials._isSecure()) {
+            this.target = new url.URL(`https://${address}`);
+        }
+        else {
+            this.target = new url.URL(`http://${address}`);
+        }
+        // TODO(murgatroid99): Add more centralized handling of channel options
+        if (this.options['grpc.default_authority']) {
+            this.defaultAuthority = this.options['grpc.default_authority'];
+        }
+        else {
+            this.defaultAuthority = this.target.host;
+        }
+        this.filterStackFactory = new filter_stack_1.FilterStackFactory([
+            new call_credentials_filter_1.CallCredentialsFilterFactory(this), new deadline_filter_1.DeadlineFilterFactory(this),
+            new metadata_status_filter_1.MetadataStatusFilterFactory(this), new compression_filter_1.CompressionFilterFactory(this)
+        ]);
+        this.currentBackoffDeadline = new Date();
+        /* The only purpose of these lines is to ensure that this.backoffTimerId has
+         * a value of type NodeJS.Timer. */
+        this.backoffTimerId = setTimeout(() => { }, 0);
+        // Build user-agent string.
+        this.userAgent = [
+            options['grpc.primary_user_agent'], `grpc-node-js/${clientVersion}`,
+            options['grpc.secondary_user_agent']
+        ].filter(e => e).join(' '); // remove falsey values first
+    }
+    handleStateChange(oldState, newState) {
+        const now = new Date();
+        switch (newState) {
+            case ConnectivityState.CONNECTING:
+                if (oldState === ConnectivityState.IDLE) {
+                    this.currentBackoff = INITIAL_BACKOFF_MS;
+                    this.currentBackoffDeadline =
+                        new Date(now.getTime() + INITIAL_BACKOFF_MS);
+                }
+                else if (oldState === ConnectivityState.TRANSIENT_FAILURE) {
+                    this.currentBackoff = Math.min(this.currentBackoff * BACKOFF_MULTIPLIER, MAX_BACKOFF_MS);
+                    const jitterMagnitude = BACKOFF_JITTER * this.currentBackoff;
+                    this.currentBackoffDeadline = new Date(now.getTime() + this.currentBackoff +
+                        uniformRandom(-jitterMagnitude, jitterMagnitude));
+                }
+                this.startConnecting();
+                break;
+            case ConnectivityState.READY:
+                this.emit('connect');
+                break;
+            case ConnectivityState.TRANSIENT_FAILURE:
+                this.subChannel = null;
+                this.backoffTimerId = setTimeout(() => {
+                    this.transitionToState([ConnectivityState.TRANSIENT_FAILURE], ConnectivityState.CONNECTING);
+                }, this.currentBackoffDeadline.getTime() - now.getTime());
+                break;
+            case ConnectivityState.IDLE:
+            case ConnectivityState.SHUTDOWN:
+                if (this.subChannel) {
+                    this.subChannel.close();
+                    this.subChannel.removeListener('connect', this.subChannelConnectCallback);
+                    this.subChannel.removeListener('close', this.subChannelCloseCallback);
+                    this.subChannel = null;
+                    this.emit('shutdown');
+                    clearTimeout(this.backoffTimerId);
+                }
+                break;
+            default:
+                throw new Error('This should never happen');
+        }
+    }
+    // Transition from any of a set of oldStates to a specific newState
+    transitionToState(oldStates, newState) {
+        if (oldStates.indexOf(this.connectivityState) > -1) {
+            const oldState = this.connectivityState;
+            this.connectivityState = newState;
+            this.handleStateChange(oldState, newState);
+            this.emit('connectivityStateChanged', newState);
+        }
+    }
+    startConnecting() {
+        const connectionOptions = this.credentials._getConnectionOptions() || {};
+        if (connectionOptions.secureContext !== null) {
+            // If provided, the value of grpc.ssl_target_name_override should be used
+            // to override the target hostname when checking server identity.
+            // This option is used for testing only.
+            if (this.options['grpc.ssl_target_name_override']) {
+                const sslTargetNameOverride = this.options['grpc.ssl_target_name_override'];
+                connectionOptions.checkServerIdentity =
+                    (host, cert) => {
+                        return tls_1.checkServerIdentity(sslTargetNameOverride, cert);
+                    };
+                connectionOptions.servername = sslTargetNameOverride;
+            }
+        }
+        const subChannel = new subchannel_1.Http2SubChannel(this.target, connectionOptions, this.userAgent, this.options);
+        this.subChannel = subChannel;
+        const now = new Date();
+        const connectionTimeout = Math.max(this.currentBackoffDeadline.getTime() - now.getTime(), MIN_CONNECT_TIMEOUT_MS);
+        const connectionTimerId = setTimeout(() => {
+            // This should trigger the 'close' event, which will send us back to
+            // TRANSIENT_FAILURE
+            subChannel.close();
+        }, connectionTimeout);
+        this.subChannelConnectCallback = () => {
+            // Connection succeeded
+            clearTimeout(connectionTimerId);
+            this.transitionToState([ConnectivityState.CONNECTING], ConnectivityState.READY);
+        };
+        subChannel.once('connect', this.subChannelConnectCallback);
+        this.subChannelCloseCallback = () => {
+            // Connection failed
+            clearTimeout(connectionTimerId);
+            /* TODO(murgatroid99): verify that this works for
+             * CONNECTING->TRANSITIVE_FAILURE see nodejs/node#16645 */
+            this.transitionToState([ConnectivityState.CONNECTING, ConnectivityState.READY], ConnectivityState.TRANSIENT_FAILURE);
+        };
+        subChannel.once('close', this.subChannelCloseCallback);
+    }
+    _startHttp2Stream(authority, methodName, stream, metadata) {
+        const finalMetadata = stream.filterStack.sendMetadata(Promise.resolve(metadata.clone()));
+        Promise.all([finalMetadata, this.connect()])
+            .then(([metadataValue]) => {
+            const headers = metadataValue.toHttp2Headers();
+            headers[HTTP2_HEADER_AUTHORITY] = authority;
+            headers[HTTP2_HEADER_USER_AGENT] = this.userAgent;
+            headers[HTTP2_HEADER_CONTENT_TYPE] = 'application/grpc';
+            headers[HTTP2_HEADER_METHOD] = 'POST';
+            headers[HTTP2_HEADER_PATH] = methodName;
+            headers[HTTP2_HEADER_TE] = 'trailers';
+            if (this.connectivityState === ConnectivityState.READY) {
+                const subChannel = this.subChannel;
+                subChannel.startCallStream(metadataValue, stream);
+            }
+            else {
+                /* In this case, we lost the connection while finalizing
+                 * metadata. That should be very unusual */
+                setImmediate(() => {
+                    this._startHttp2Stream(authority, methodName, stream, metadata);
+                });
+            }
+        })
+            .catch((error) => {
+            // We assume the error code isn't 0 (Status.OK)
+            stream.cancelWithStatus(error.code || constants_1.Status.UNKNOWN, `Getting metadata from plugin failed with error: ${error.message}`);
+        });
+    }
+    createCall(method, deadline, host, parentCall, propagateFlags) {
+        if (this.connectivityState === ConnectivityState.SHUTDOWN) {
+            throw new Error('Channel has been shut down');
+        }
+        const finalOptions = {
+            deadline: (deadline === null || deadline === undefined) ? Infinity :
+                deadline,
+            flags: propagateFlags || 0,
+            host: host || this.defaultAuthority,
+            parentCall: parentCall || null
+        };
+        const stream = new call_stream_1.Http2CallStream(method, this, finalOptions, this.filterStackFactory);
+        return stream;
+    }
+    /**
+     * Attempts to connect, returning a Promise that resolves when the connection
+     * is successful, or rejects if the channel is shut down.
+     */
+    connect() {
+        if (this.connectivityState === ConnectivityState.READY) {
+            return Promise.resolve();
+        }
+        else if (this.connectivityState === ConnectivityState.SHUTDOWN) {
+            return Promise.reject(new Error('Channel has been shut down'));
+        }
+        else {
+            // In effect, this.connecting is only assigned upon the first attempt to
+            // transition from IDLE to CONNECTING, so this condition could have also
+            // been (connectivityState === IDLE).
+            if (!this.connecting) {
+                this.connecting = new Promise((resolve, reject) => {
+                    this.transitionToState([ConnectivityState.IDLE], ConnectivityState.CONNECTING);
+                    const onConnect = () => {
+                        this.connecting = null;
+                        this.removeListener('shutdown', onShutdown);
+                        resolve();
+                    };
+                    const onShutdown = () => {
+                        this.connecting = null;
+                        this.removeListener('connect', onConnect);
+                        reject(new Error('Channel has been shut down'));
+                    };
+                    this.once('connect', onConnect);
+                    this.once('shutdown', onShutdown);
+                });
+            }
+            return this.connecting;
+        }
+    }
+    getConnectivityState(tryToConnect) {
+        if (tryToConnect) {
+            this.transitionToState([ConnectivityState.IDLE], ConnectivityState.CONNECTING);
+        }
+        return this.connectivityState;
+    }
+    watchConnectivityState(currentState, deadline, callback) {
+        if (this.connectivityState !== currentState) {
+            /* If the connectivity state is different from the provided currentState,
+             * we assume that a state change has successfully occurred */
+            setImmediate(callback);
+        }
+        else {
+            let deadlineMs = 0;
+            if (deadline instanceof Date) {
+                deadlineMs = deadline.getTime();
+            }
+            else {
+                deadlineMs = deadline;
+            }
+            let timeout = deadlineMs - Date.now();
+            if (timeout < 0) {
+                timeout = 0;
+            }
+            const timeoutId = setTimeout(() => {
+                this.removeListener('connectivityStateChanged', eventCb);
+                callback(new Error('Channel state did not change before deadline'));
+            }, timeout);
+            const eventCb = () => {
+                clearTimeout(timeoutId);
+                callback();
+            };
+            this.once('connectivityStateChanged', eventCb);
+        }
+    }
+    getTarget() {
+        return this.target.toString();
+    }
+    close() {
+        if (this.connectivityState === ConnectivityState.SHUTDOWN) {
+            throw new Error('Channel has been shut down');
+        }
+        this.transitionToState([
+            ConnectivityState.CONNECTING, ConnectivityState.READY,
+            ConnectivityState.TRANSIENT_FAILURE, ConnectivityState.IDLE
+        ], ConnectivityState.SHUTDOWN);
+    }
+}
+exports.Http2Channel = Http2Channel;
+//# sourceMappingURL=channel.js.map
\ No newline at end of file