All files / rpc-calls/src/testing Callee.fixtures.ts

83.6% Statements 51/61
56.25% Branches 9/16
76.31% Functions 29/38
83.33% Lines 45/54

Press n or j to go to the next uncovered block, b, p or k for the previous block.

1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 1689x 9x 9x 9x 9x           9x       17x                       21x       2x     21x                                           78x 73x       10x       80x               3x   3x               27x               4x     24x     24x 24x 24x 24x 24x 24x 24x     24x 24x               49x 45x       25x   25x 15x               1x     1x           1x   1x         1x 1x       1x   1x 1x             9x   9x 156x      
import {timer, from, Observable, of} from 'rxjs';
import {map, switchMap, take} from 'rxjs/operators';
import {RpcError} from '@jsonjoy.com/rpc-error';
import {Procedure} from '../procedures';
import {RpcCallee} from '../callee/RpcCallee';
 
export interface SampleCtx {
  ip?: string;
}
 
export const procedures = {
  ping: Procedure.new<void, 'pong', SampleCtx>('pong'),
 
  getIp: Procedure.new<void, {ip: string}, SampleCtx>((inp, ctx: SampleCtx) => {
    return {ip: ctx.ip ?? ''};
  }),
 
  delay: Procedure.unary(async ({timeout = 10}: {timeout?: number} = {}) => {
    await new Promise((r) => setTimeout(r, timeout));
    return {
      done: true,
      timeout,
    };
  }),
 
  notificationSetValue: Procedure.new(({value}: {value: number}) => {
    valueHolder.value = value;
  }),
 
  notificationSetValueFromCtx: Procedure.new((inp, ctx: SampleCtx) => {
    valueHolder.value = ctx?.ip?.length ?? 0;
  }),
 
  getValue: Procedure.new(() => ({value: valueHolder.value})),
 
  delayStreaming: Procedure.rx<{timeout?: number}, {done: true; timeout: number}>((req$) =>
    req$.pipe(
      take(1),
      switchMap(({timeout = 10}: {timeout?: number} = {}) =>
        from(
          new Promise<number>((r) => {
            setTimeout(() => {
              r(timeout);
            }, timeout);
          }),
        ),
      ),
      map((timeout: number) => ({
        done: true,
        timeout,
      })),
    ),
  ),
 
  double: Procedure.unary(async ({num}: {num: number}) => {
    if (typeof num !== 'number') throw RpcError.validation('Payload .num field missing.');
    return {num: num * 2};
  }),
 
  error: Procedure.unary(async () => {
    throw new RpcError('this promise can throw', '', 0, '', undefined, undefined);
  }),
 
  'auth.users.get': Procedure.unary(async ({id}: {id: string}) => {
    return {
      id,
      name: 'Mario Dragi',
      tags: ['news', 'cola', 'bcaa'],
    };
  }),
 
  streamError: Procedure.rx(() =>
    from(
      (async () => {
        throw RpcError.internal(null, 'Stream always errors');
      })(),
    ),
  ),
 
  utilTimer: Procedure.rx(() => timer(10, 10)),
 
  'util.info': Procedure.rx(() =>
    from([
      {
        commit: 'AAAAAAAAAAAAAAAAAAA',
        sha1: 'BBBBBBBBBBBBBBBBBBB',
      },
    ]),
  ),
 
  'util.timer': Procedure.rx(() => timer(10, 10)),
 
  count: Procedure.rx<{count: number}, number>((request$) =>
    request$.pipe(
      switchMap(
        ({count}: {count: number}) =>
          new Observable<number>((observer) => {
            let cnt = 0;
            const timerId = setInterval(() => {
              observer.next(cnt++);
              Eif (cnt >= count) {
                observer.complete();
                clearInterval(timerId);
              }
            }, 10);
            return () => {
              clearInterval(timerId);
            };
          }),
      ),
    ),
  ),
 
  doubleStringWithValidation: Procedure.unary(async ({foo}: {foo: string}) => {
    if (typeof foo !== 'string') throw RpcError.validation('"foo" property missing.');
    return {bar: foo + foo};
  }),
 
  doubleStringWithValidation2: Procedure.rx<{foo: string}, {bar: string}>((req$) =>
    req$.pipe(
      map(({foo}: {foo: string}) => {
        if (typeof foo !== 'string') throw RpcError.validation('"foo" property missing.');
        return {bar: foo + foo};
      }),
    ),
  ),
 
  passthroughStream: Procedure.rx((req$) => req$),
 
  emitOnceSync: Procedure.rx((request$, ctx) => {
    return request$.pipe(
      take(1),
      switchMap(async (request) => {
        return JSON.stringify({request, ctx});
      }),
    );
  }),
 
  emitThreeSync: Procedure.rx((request$) => {
    return request$.pipe(
      take(1),
      switchMap(() => from([1, 2, 3])),
    );
  }),
 
  promiseDelay: Procedure.unary(async () => {
    await new Promise((r) => setTimeout(r, 5));
    return {};
  }),
 
  streamDelay: Procedure.rx(() => {
    return of({}).pipe(
      switchMap(async () => {
        await new Promise((r) => setTimeout(r, 5));
        return {};
      }),
    );
  }),
};
 
// Helper for value state
const valueHolder = {value: 0};
 
export const createRpcCallee = <Ctx>() =>
  new RpcCallee<Ctx, typeof procedures>({
    procedures,
  });