All files / sportident/src/SiCard raceResultTools.ts

100% Statements 66/66
100% Branches 53/53
100% Functions 16/16
100% Lines 60/60

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 168 169 170 171 172 173 17418x                                     18x 6x     2x           6x     2x           6x                       18x     10x         10x 5x 5x   10x 6x 6x   10x 6x 6x   10x 6x 6x 9x 7x       7x   2x           6x   10x 6x 6x   10x     18x     10x 9x   10x 40x   10x                 9x               18x 6x 6x 6x   22x 8x   22x 22x     6x           18x 5x 5x 5x     18x     6x 6x 2x   4x 5x   4x 16x   4x               5x              
import * as siProtocol from '../siProtocol';
import {IRaceResultData, IPunch} from './IRaceResultData';
 
export interface IOrderedRaceResult {
    cardNumber?: number;
    cardHolder?: {[property: string]: unknown};
    orderedTimes: number[];
    clearTimeIndex?: number|null;
    checkTimeIndex?: number|null;
    startTimeIndex?: number|null;
    punches?: IOrderedPunch[];
    finishTimeIndex?: number|null;
}
 
export interface IOrderedPunch {
    code: number;
    timeIndex: number|null;
}
 
export const prettyRaceResult = (result: IRaceResultData): string => {
    const punchesString = (result.punches
        ? (result.punches.length > 0
            ? result.punches.map(
                (punch) => `${punch.code}: ${punch.time}`,
            ).join('\n')
            : 'No Punches'
        )
        : '? Punches'
    );
    const cardHolderString = (result.cardHolder
        ? (Object.keys(result.cardHolder).length > 0
            ? Object.keys(result.cardHolder).map(
                (key) => `${key}: ${result.cardHolder![key]}`,
            ).join('\n')
            : 'Empty Card Holder'
        )
        : '?'
    );
    return (
        `Card Number: ${result.cardNumber !== undefined ? result.cardNumber : '?'}\n` +
        `Clear: ${result.clearTime !== undefined ? result.clearTime : '?'}\n` +
        `Check: ${result.checkTime !== undefined ? result.checkTime : '?'}\n` +
        `Start: ${result.startTime !== undefined ? result.startTime : '?'}\n` +
        `Finish: ${result.finishTime !== undefined ? result.finishTime : '?'}\n` +
        `${punchesString}\n` +
        'Card Holder:\n' +
        `${cardHolderString}\n`
    );
};
 
export const getOrderedRaceResult = (
    result: IRaceResultData,
): IOrderedRaceResult => {
    const orderedResult: IOrderedRaceResult = {
        cardNumber: result.cardNumber,
        cardHolder: result.cardHolder,
        orderedTimes: [],
    };
    if (result.clearTime !== undefined && result.clearTime !== null) {
        orderedResult.clearTimeIndex = orderedResult.orderedTimes.length;
        orderedResult.orderedTimes.push(result.clearTime);
    }
    if (result.checkTime !== undefined && result.checkTime !== null) {
        orderedResult.checkTimeIndex = orderedResult.orderedTimes.length;
        orderedResult.orderedTimes.push(result.checkTime);
    }
    if (result.startTime !== undefined && result.startTime !== null) {
        orderedResult.startTimeIndex = orderedResult.orderedTimes.length;
        orderedResult.orderedTimes.push(result.startTime);
    }
    if (result.punches !== undefined) {
        const newPunches: IOrderedPunch[] = [];
        result.punches.forEach((punch: IPunch) => {
            if (punch.time !== null) {
                newPunches.push({
                    code: punch.code,
                    timeIndex: orderedResult.orderedTimes.length,
                });
                orderedResult.orderedTimes.push(punch.time);
            } else {
                newPunches.push({
                    code: punch.code,
                    timeIndex: null,
                });
            }
        });
        orderedResult.punches = newPunches;
    }
    if (result.finishTime !== undefined && result.finishTime !== null) {
        orderedResult.finishTimeIndex = orderedResult.orderedTimes.length;
        orderedResult.orderedTimes.push(result.finishTime);
    }
    return orderedResult;
};
 
export const getRaceResultFromOrdered = (
    orderedResult: IOrderedRaceResult,
): IRaceResultData => {
    const getOrderedTimeIndexIfSet = (index: number|null) => (
        index === null ? null : orderedResult.orderedTimes[index]
    );
    const getOrderedTimeIndexIfSetAndDefined = (index: number|null|undefined) => (
        index === undefined || index === null ? index : orderedResult.orderedTimes[index]
    );
    return {
        cardNumber: orderedResult.cardNumber,
        cardHolder: orderedResult.cardHolder,
        clearTime: getOrderedTimeIndexIfSetAndDefined(orderedResult.clearTimeIndex),
        checkTime: getOrderedTimeIndexIfSetAndDefined(orderedResult.checkTimeIndex),
        startTime: getOrderedTimeIndexIfSetAndDefined(orderedResult.startTimeIndex),
        finishTime: getOrderedTimeIndexIfSetAndDefined(orderedResult.finishTimeIndex),
        punches: (orderedResult.punches === undefined
            ? undefined
            : orderedResult.punches.map((punch: IOrderedPunch) => ({
                code: punch.code,
                time: getOrderedTimeIndexIfSet(punch.timeIndex),
            }))
        ),
    };
};
 
export const monotonizeOrderedRaceResult = (orderedData: IOrderedRaceResult): IOrderedRaceResult => {
    let currentCarry = 0;
    let lastTime = 0;
    const newOrderedTimes = orderedData.orderedTimes.map(
        (time: number) => {
            if (time < lastTime) {
                currentCarry += siProtocol.SI_TIME_CUTOFF;
            }
            lastTime = time;
            return time + currentCarry;
        },
    );
    return {
        ...orderedData,
        orderedTimes: newOrderedTimes,
    };
};
 
export const monotonizeRaceResult = (result: IRaceResultData): IRaceResultData => {
    const orderedResult = getOrderedRaceResult(result);
    const monotonizedOrderedResult = monotonizeOrderedRaceResult(orderedResult);
    return getRaceResultFromOrdered(monotonizedOrderedResult);
};
 
export const makeStartZeroTime = (
    result: IRaceResultData,
): IRaceResultData => {
    const zeroTime = result.startTime;
    if (zeroTime === undefined || zeroTime === null) {
        throw new Error('start time must be known');
    }
    const deductZeroTimeIfSet = (time: siProtocol.SiTimestamp) => (
        time === null ? null : time - zeroTime
    );
    const deductZeroTimeIfSetAndDefined = (time: siProtocol.SiTimestamp|undefined) => (
        time === null || time === undefined ? time : time - zeroTime
    );
    return {
        ...result,
        clearTime: deductZeroTimeIfSetAndDefined(result.clearTime),
        checkTime: deductZeroTimeIfSetAndDefined(result.checkTime),
        startTime: deductZeroTimeIfSetAndDefined(result.startTime),
        finishTime: deductZeroTimeIfSetAndDefined(result.finishTime),
        punches: (result.punches === undefined
            ? undefined
            : result.punches.map((punch: IPunch) => ({
                ...punch,
                time: deductZeroTimeIfSet(punch.time),
            }))
        ),
    };
};