| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114 |
- import { describe, expect, it } from 'vitest';
- import { syncObservatories } from '@/pages/xray/balancers/balancer-helpers';
- import type { XraySettingsValue } from '@/hooks/useXraySetting';
- function tpl(routing: Record<string, unknown>, extra: Record<string, unknown> = {}): XraySettingsValue {
- return { routing, ...extra } as unknown as XraySettingsValue;
- }
- // Observatory sections have no reload API in xray-core, so creating one turns
- // a balancer save from a live (hot-applied) routing change into a full
- // restart. These tests pin the rule: only strategies that genuinely need an
- // observer may create one — which, for random/roundRobin, means a fallbackTag
- // is set (xray-core then requires the Observatory feature; see #5605).
- describe('syncObservatories', () => {
- it('does not create burstObservatory for a fresh random balancer (stays hot-appliable)', () => {
- const t = tpl({ balancers: [{ tag: 'b1', selector: ['direct'] }] });
- syncObservatories(t);
- expect(t.burstObservatory).toBeUndefined();
- expect(t.observatory).toBeUndefined();
- });
- it('does not create burstObservatory for roundRobin without fallback', () => {
- const t = tpl({ balancers: [{ tag: 'b1', selector: ['a'], strategy: { type: 'roundRobin' } }] });
- syncObservatories(t);
- expect(t.burstObservatory).toBeUndefined();
- });
- it('creates burstObservatory for a random balancer with a fallbackTag (#5605)', () => {
- const t = tpl({ balancers: [{ tag: 'OverProxy', selector: ['opera-proxy'], fallbackTag: 'warp' }] });
- syncObservatories(t);
- expect(t.burstObservatory).toBeDefined();
- expect((t.burstObservatory as { subjectSelector: string[] }).subjectSelector).toEqual(['opera-proxy']);
- });
- it('creates burstObservatory for roundRobin with a fallbackTag', () => {
- const t = tpl({ balancers: [{ tag: 'b1', selector: ['a'], fallbackTag: 'warp', strategy: { type: 'roundRobin' } }] });
- syncObservatories(t);
- expect(t.burstObservatory).toBeDefined();
- expect((t.burstObservatory as { subjectSelector: string[] }).subjectSelector).toEqual(['a']);
- });
- it('treats an empty-string fallbackTag as no fallback (stays hot-appliable)', () => {
- const t = tpl({ balancers: [{ tag: 'b1', selector: ['a'], fallbackTag: '' }] });
- syncObservatories(t);
- expect(t.burstObservatory).toBeUndefined();
- });
- it('removes burstObservatory when a random balancer drops its fallbackTag', () => {
- const t = tpl(
- { balancers: [{ tag: 'OverProxy', selector: ['opera-proxy'], fallbackTag: '' }] },
- { burstObservatory: { subjectSelector: ['opera-proxy'] } },
- );
- syncObservatories(t);
- expect(t.burstObservatory).toBeUndefined();
- });
- it('removes burstObservatory when a roundRobin balancer drops its fallbackTag', () => {
- const t = tpl(
- { balancers: [{ tag: 'b1', selector: ['a'], strategy: { type: 'roundRobin' } }] },
- { burstObservatory: { subjectSelector: ['a'] } },
- );
- syncObservatories(t);
- expect(t.burstObservatory).toBeUndefined();
- });
- it('keeps burstObservatory while another fallback balancer still needs it', () => {
- const t = tpl(
- {
- balancers: [
- { tag: 'b1', selector: ['a'] },
- { tag: 'b2', selector: ['b'], fallbackTag: 'warp', strategy: { type: 'roundRobin' } },
- ],
- },
- { burstObservatory: { subjectSelector: ['a', 'b'] } },
- );
- syncObservatories(t);
- expect(t.burstObservatory).toBeDefined();
- expect((t.burstObservatory as { subjectSelector: string[] }).subjectSelector).toEqual(['b', 'a']);
- });
- it('creates burstObservatory for leastLoad (required by the strategy)', () => {
- const t = tpl({ balancers: [{ tag: 'b1', selector: ['a'], strategy: { type: 'leastLoad' } }] });
- syncObservatories(t);
- expect(t.burstObservatory).toBeDefined();
- expect((t.burstObservatory as { subjectSelector: string[] }).subjectSelector).toEqual(['a']);
- });
- it('creates observatory for leastPing (required by the strategy)', () => {
- const t = tpl({ balancers: [{ tag: 'b1', selector: ['a'], strategy: { type: 'leastPing' } }] });
- syncObservatories(t);
- expect(t.observatory).toBeDefined();
- expect((t.observatory as { subjectSelector: string[] }).subjectSelector).toEqual(['a']);
- });
- it('keeps an existing burstObservatory in sync for random balancers (legacy setups)', () => {
- const t = tpl(
- { balancers: [{ tag: 'b1', selector: ['a'] }, { tag: 'b2', selector: ['b'], strategy: { type: 'leastLoad' } }] },
- { burstObservatory: { subjectSelector: ['stale'] } },
- );
- syncObservatories(t);
- expect((t.burstObservatory as { subjectSelector: string[] }).subjectSelector).toEqual(['b', 'a']);
- });
- it('removes observatories when no balancer can use them', () => {
- const t = tpl({ balancers: [] }, {
- observatory: { subjectSelector: ['a'] },
- burstObservatory: { subjectSelector: ['a'] },
- });
- syncObservatories(t);
- expect(t.observatory).toBeUndefined();
- expect(t.burstObservatory).toBeUndefined();
- });
- });
|