mirror of
https://github.com/swc-project/swc.git
synced 2024-12-01 09:52:57 +03:00
219 lines
5.4 KiB
TypeScript
219 lines
5.4 KiB
TypeScript
// Loaded from https://raw.githubusercontent.com/nats-io/nats.deno/v1.0.0-rc4/nats-base-client/servers.ts
|
|
|
|
|
|
/*
|
|
* Copyright 2018-2021 The NATS Authors
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
* you may not use this file except in compliance with the License.
|
|
* You may obtain a copy of the License at
|
|
*
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
*
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
* See the License for the specific language governing permissions and
|
|
* limitations under the License.
|
|
*
|
|
*/
|
|
import {
|
|
DEFAULT_HOST,
|
|
DEFAULT_PORT,
|
|
Server,
|
|
ServerInfo,
|
|
ServersChanged,
|
|
URLParseFn,
|
|
} from "./types.ts";
|
|
import { defaultPort, getUrlParseFn } from "./transport.ts";
|
|
import { shuffle } from "./util.ts";
|
|
import { isIP } from "./ipparser.ts";
|
|
|
|
/**
|
|
* @hidden
|
|
*/
|
|
export class ServerImpl implements Server {
|
|
src: string;
|
|
listen: string;
|
|
hostname: string;
|
|
port: number;
|
|
didConnect: boolean;
|
|
reconnects: number;
|
|
lastConnect: number;
|
|
gossiped: boolean;
|
|
tlsName: string;
|
|
|
|
constructor(u: string, gossiped = false) {
|
|
this.src = u;
|
|
this.tlsName = "";
|
|
// remove any protocol that may have been provided
|
|
if (u.match(/^(.*:\/\/)(.*)/m)) {
|
|
u = u.replace(/^(.*:\/\/)(.*)/gm, "$2");
|
|
}
|
|
// in web environments, URL may not be a living standard
|
|
// that means that protocols other than HTTP/S are not
|
|
// parsable correctly.
|
|
const url = new URL(`http://${u}`);
|
|
if (!url.port) {
|
|
url.port = `${DEFAULT_PORT}`;
|
|
}
|
|
this.listen = url.host;
|
|
this.hostname = url.hostname;
|
|
this.port = parseInt(url.port, 10);
|
|
|
|
this.didConnect = false;
|
|
this.reconnects = 0;
|
|
this.lastConnect = 0;
|
|
this.gossiped = gossiped;
|
|
}
|
|
|
|
toString(): string {
|
|
return this.listen;
|
|
}
|
|
}
|
|
|
|
export interface ServersOptions {
|
|
urlParseFn?: URLParseFn;
|
|
}
|
|
|
|
/**
|
|
* @hidden
|
|
*/
|
|
export class Servers {
|
|
private firstSelect: boolean;
|
|
private readonly servers: ServerImpl[];
|
|
private currentServer: ServerImpl;
|
|
private tlsName: string;
|
|
|
|
constructor(
|
|
randomize: boolean,
|
|
listens: string[] = [],
|
|
opts: ServersOptions = {},
|
|
) {
|
|
this.firstSelect = true;
|
|
this.servers = [] as ServerImpl[];
|
|
this.tlsName = "";
|
|
|
|
const urlParseFn = getUrlParseFn();
|
|
if (listens) {
|
|
listens.forEach((hp) => {
|
|
hp = urlParseFn ? urlParseFn(hp) : hp;
|
|
this.servers.push(new ServerImpl(hp));
|
|
});
|
|
if (randomize) {
|
|
this.servers = shuffle(this.servers);
|
|
}
|
|
}
|
|
if (this.servers.length === 0) {
|
|
this.addServer(`${DEFAULT_HOST}:${defaultPort()}`, false);
|
|
}
|
|
this.currentServer = this.servers[0];
|
|
}
|
|
|
|
updateTLSName(): void {
|
|
const cs = this.getCurrentServer();
|
|
if (!isIP(cs.hostname)) {
|
|
this.tlsName = cs.hostname;
|
|
this.servers.forEach((s) => {
|
|
if (s.gossiped) {
|
|
s.tlsName = this.tlsName;
|
|
}
|
|
});
|
|
}
|
|
}
|
|
|
|
getCurrentServer(): ServerImpl {
|
|
return this.currentServer;
|
|
}
|
|
|
|
addServer(u: string, implicit = false): void {
|
|
const urlParseFn = getUrlParseFn();
|
|
u = urlParseFn ? urlParseFn(u) : u;
|
|
const s = new ServerImpl(u, implicit);
|
|
if (isIP(s.hostname)) {
|
|
s.tlsName = this.tlsName;
|
|
}
|
|
this.servers.push(s);
|
|
}
|
|
|
|
selectServer(): ServerImpl | undefined {
|
|
// allow using select without breaking the order of the servers
|
|
if (this.firstSelect) {
|
|
this.firstSelect = false;
|
|
return this.currentServer;
|
|
}
|
|
const t = this.servers.shift();
|
|
if (t) {
|
|
this.servers.push(t);
|
|
this.currentServer = t;
|
|
}
|
|
return t;
|
|
}
|
|
|
|
removeCurrentServer(): void {
|
|
this.removeServer(this.currentServer);
|
|
}
|
|
|
|
removeServer(server: ServerImpl | undefined): void {
|
|
if (server) {
|
|
const index = this.servers.indexOf(server);
|
|
this.servers.splice(index, 1);
|
|
}
|
|
}
|
|
|
|
length(): number {
|
|
return this.servers.length;
|
|
}
|
|
|
|
next(): ServerImpl | undefined {
|
|
return this.servers.length ? this.servers[0] : undefined;
|
|
}
|
|
|
|
getServers(): ServerImpl[] {
|
|
return this.servers;
|
|
}
|
|
|
|
update(info: ServerInfo): ServersChanged {
|
|
const added: string[] = [];
|
|
let deleted: string[] = [];
|
|
|
|
const urlParseFn = getUrlParseFn();
|
|
const discovered = new Map<string, ServerImpl>();
|
|
if (info.connect_urls && info.connect_urls.length > 0) {
|
|
info.connect_urls.forEach((hp) => {
|
|
hp = urlParseFn ? urlParseFn(hp) : hp;
|
|
const s = new ServerImpl(hp, true);
|
|
discovered.set(hp, s);
|
|
});
|
|
}
|
|
// remove gossiped servers that are no longer reported
|
|
const toDelete: number[] = [];
|
|
this.servers.forEach((s, index) => {
|
|
const u = s.listen;
|
|
if (
|
|
s.gossiped && this.currentServer.listen !== u &&
|
|
discovered.get(u) === undefined
|
|
) {
|
|
// server was removed
|
|
toDelete.push(index);
|
|
}
|
|
// remove this entry from reported
|
|
discovered.delete(u);
|
|
});
|
|
|
|
// perform the deletion
|
|
toDelete.reverse();
|
|
toDelete.forEach((index) => {
|
|
const removed = this.servers.splice(index, 1);
|
|
deleted = deleted.concat(removed[0].listen);
|
|
});
|
|
|
|
// remaining servers are new
|
|
discovered.forEach((v, k, m) => {
|
|
this.servers.push(v);
|
|
added.push(k);
|
|
});
|
|
|
|
return { added, deleted };
|
|
}
|
|
}
|