mirror of
https://github.com/spacebarchat/server.git
synced 2024-11-11 05:02:37 +01:00
Merge pull request #297 from AlTech98/typeorm
Implemented voice apis #127 and #78
This commit is contained in:
commit
372142852b
@ -1,10 +1,15 @@
|
|||||||
import { Config } from "@fosscord/util";
|
import {Config, Guild, Member} from "@fosscord/util";
|
||||||
import { Request, Response, Router } from "express";
|
import { Request, Response, Router } from "express";
|
||||||
|
import {getVoiceRegions} from "../../../util/Voice";
|
||||||
|
import {getIpAdress} from "../../../util/ipAddress";
|
||||||
|
|
||||||
const router = Router();
|
const router = Router();
|
||||||
|
|
||||||
router.get("/", async (req: Request, res: Response) => {
|
router.get("/", async (req: Request, res: Response) => {
|
||||||
return res.json(Config.get().regions.available);
|
const { guild_id } = req.params;
|
||||||
|
const guild = await Guild.findOneOrFail({ id: guild_id });
|
||||||
|
//TODO we should use an enum for guild's features and not hardcoded strings
|
||||||
|
return res.json(await getVoiceRegions(getIpAdress(req), guild.features.includes("VIP_REGIONS")));
|
||||||
});
|
});
|
||||||
|
|
||||||
export default router;
|
export default router;
|
||||||
|
11
api/src/routes/voice/regions.ts
Normal file
11
api/src/routes/voice/regions.ts
Normal file
@ -0,0 +1,11 @@
|
|||||||
|
import { Router, Request, Response } from "express";
|
||||||
|
import {getIpAdress} from "../../util/ipAddress";
|
||||||
|
import {getVoiceRegions} from "../../util/Voice";
|
||||||
|
|
||||||
|
const router: Router = Router();
|
||||||
|
|
||||||
|
router.get("/", async (req: Request, res: Response) => {
|
||||||
|
res.json(await getVoiceRegions(getIpAdress(req), true))//vip true?
|
||||||
|
});
|
||||||
|
|
||||||
|
export default router;
|
32
api/src/util/Voice.ts
Normal file
32
api/src/util/Voice.ts
Normal file
@ -0,0 +1,32 @@
|
|||||||
|
import {Config} from "@fosscord/util";
|
||||||
|
import {distanceBetweenLocations, IPAnalysis} from "./ipAddress";
|
||||||
|
|
||||||
|
export async function getVoiceRegions(ipAddress: string, vip: boolean) {
|
||||||
|
const regions = Config.get().regions;
|
||||||
|
const availableRegions = regions.available.filter(ar => vip ? true : !ar.vip);
|
||||||
|
let optimalId = regions.default
|
||||||
|
|
||||||
|
if(!regions.useDefaultAsOptimal) {
|
||||||
|
const clientIpAnalysis = await IPAnalysis(ipAddress)
|
||||||
|
|
||||||
|
let min = Number.POSITIVE_INFINITY
|
||||||
|
|
||||||
|
for (let ar of availableRegions) {
|
||||||
|
//TODO the endpoint location should be saved in the database if not already present to prevent IPAnalysis call
|
||||||
|
const dist = distanceBetweenLocations(clientIpAnalysis, ar.location || (await IPAnalysis(ar.endpoint)))
|
||||||
|
|
||||||
|
if(dist < min) {
|
||||||
|
min = dist
|
||||||
|
optimalId = ar.id
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return availableRegions.map(ar => ({
|
||||||
|
id: ar.id,
|
||||||
|
name: ar.name,
|
||||||
|
custom: ar.custom,
|
||||||
|
deprecated: ar.deprecated,
|
||||||
|
optimal: ar.id === optimalId
|
||||||
|
}))
|
||||||
|
}
|
@ -60,6 +60,7 @@ const exampleData = {
|
|||||||
status: 200
|
status: 200
|
||||||
};
|
};
|
||||||
|
|
||||||
|
//TODO add function that support both ip and domain names
|
||||||
export async function IPAnalysis(ip: string): Promise<typeof exampleData> {
|
export async function IPAnalysis(ip: string): Promise<typeof exampleData> {
|
||||||
const { ipdataApiKey } = Config.get().security;
|
const { ipdataApiKey } = Config.get().security;
|
||||||
if (!ipdataApiKey) return { ...exampleData, ip };
|
if (!ipdataApiKey) return { ...exampleData, ip };
|
||||||
@ -79,3 +80,19 @@ export function getIpAdress(req: Request): string {
|
|||||||
// @ts-ignore
|
// @ts-ignore
|
||||||
return req.headers[Config.get().security.forwadedFor] || req.socket.remoteAddress;
|
return req.headers[Config.get().security.forwadedFor] || req.socket.remoteAddress;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
export function distanceBetweenLocations(loc1: any, loc2: any): number {
|
||||||
|
return distanceBetweenCoords(loc1.latitude, loc1.longitude, loc2.latitude, loc2.longitude);
|
||||||
|
}
|
||||||
|
|
||||||
|
//Haversine function
|
||||||
|
function distanceBetweenCoords(lat1: number, lon1: number, lat2: number, lon2: number) {
|
||||||
|
const p = 0.017453292519943295; // Math.PI / 180
|
||||||
|
const c = Math.cos;
|
||||||
|
const a = 0.5 - c((lat2 - lat1) * p) / 2 +
|
||||||
|
c(lat1 * p) * c(lat2 * p) *
|
||||||
|
(1 - c((lon2 - lon1) * p)) / 2;
|
||||||
|
|
||||||
|
return 12742 * Math.asin(Math.sqrt(a)); // 2 * R; R = 6371 km
|
||||||
|
}
|
@ -19,10 +19,14 @@ export interface RateLimitOptions {
|
|||||||
export interface Region {
|
export interface Region {
|
||||||
id: string;
|
id: string;
|
||||||
name: string;
|
name: string;
|
||||||
|
endpoint: string;
|
||||||
|
location?: {
|
||||||
|
latitude: number;
|
||||||
|
longitude: number;
|
||||||
|
};
|
||||||
vip: boolean;
|
vip: boolean;
|
||||||
custom: boolean;
|
custom: boolean;
|
||||||
deprecated: boolean;
|
deprecated: boolean;
|
||||||
optimal: boolean;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface KafkaBroker {
|
export interface KafkaBroker {
|
||||||
@ -128,6 +132,7 @@ export interface ConfigValue {
|
|||||||
};
|
};
|
||||||
regions: {
|
regions: {
|
||||||
default: string;
|
default: string;
|
||||||
|
useDefaultAsOptimal: boolean;
|
||||||
available: Region[];
|
available: Region[];
|
||||||
};
|
};
|
||||||
rabbitmq: {
|
rabbitmq: {
|
||||||
@ -263,7 +268,8 @@ export const DefaultConfigOptions: ConfigValue = {
|
|||||||
},
|
},
|
||||||
regions: {
|
regions: {
|
||||||
default: "fosscord",
|
default: "fosscord",
|
||||||
available: [{ id: "fosscord", name: "Fosscord", vip: false, custom: false, deprecated: false, optimal: false }],
|
useDefaultAsOptimal: true,
|
||||||
|
available: [{ id: "fosscord", name: "Fosscord", endpoint: "127.0.0.1", vip: false, custom: false, deprecated: false }],
|
||||||
},
|
},
|
||||||
rabbitmq: {
|
rabbitmq: {
|
||||||
host: null,
|
host: null,
|
||||||
|
@ -41,7 +41,7 @@ export class Guild extends BaseClass {
|
|||||||
explicit_content_filter?: number;
|
explicit_content_filter?: number;
|
||||||
|
|
||||||
@Column({ type: "simple-array" })
|
@Column({ type: "simple-array" })
|
||||||
features: string[];
|
features: string[]; //TODO use enum
|
||||||
|
|
||||||
@Column({ nullable: true })
|
@Column({ nullable: true })
|
||||||
icon?: string;
|
icon?: string;
|
||||||
|
Loading…
Reference in New Issue
Block a user