scottohara/loot

View on GitHub
src/securities/models/security.ts

Summary

Maintainability
A
0 mins
Test Coverage
A
100%
import type { Cacheable, Favouritable, Persistable } from "~/loot/types";
import type { OgCacheEntry } from "~/og-components/og-lru-cache-factory/types";
import type OgLruCache from "~/og-components/og-lru-cache-factory/models/og-lru-cache";
import type OgLruCacheFactory from "~/og-components/og-lru-cache-factory/models/og-lru-cache-factory";
import type { Security } from "~/securities/types";
import type { StoredAccountType } from "~/accounts/types";
import type { Transaction } from "~/transactions/types";

// Number of securities to keep in the LRU cache
const LRU_CAPACITY = 10;

export default class SecurityModel
    implements Cacheable<Security>, Favouritable<Security>, Persistable<Security>
{
    public recent: OgCacheEntry[];

    public readonly LRU_LOCAL_STORAGE_KEY = "lootRecentSecurities";

    public readonly type = "security";

    private readonly cache: angular.ICacheObject;

    private readonly lruCache: OgLruCache;

    public constructor(
        private readonly $http: angular.IHttpService,
        $cacheFactory: angular.ICacheFactoryService,
        private readonly $window: angular.IWindowService,
        ogLruCacheFactory: OgLruCacheFactory,
    ) {
        this.cache = $cacheFactory("securities");

        // Create an LRU cache and populate with the recent payee list from local storage
        this.lruCache = ogLruCacheFactory.new(LRU_CAPACITY, this.recentSecurities);
        this.recent = this.lruCache.list;
    }

    private get recentSecurities(): OgCacheEntry[] {
        const recentSecurities: string | null = this.$window.localStorage.getItem(
            this.LRU_LOCAL_STORAGE_KEY,
        );

        return JSON.parse(recentSecurities ?? "[]") as OgCacheEntry[];
    }

    // Returns the API path
    public path(id?: number): string {
        return `/securities${undefined === id ? "" : `/${id}`}`;
    }

    // Retrieves the list of securities
    public all(includeBalances = false): angular.IPromise<Security[]> {
        return this.$http
            .get(`${this.path()}${includeBalances ? "?include_balances" : ""}`, {
                cache: includeBalances ? false : this.cache,
            })
            .then(
                (response: angular.IHttpResponse<Security[]>): Security[] =>
                    response.data,
            );
    }

    // Retrieves the list of securities, including balances
    public allWithBalances(): angular.IPromise<Security[]> {
        return this.all(true);
    }

    // Retrieves the most recent transaction for a security
    public findLastTransaction(
        securityId: number,
        accountType: StoredAccountType,
    ): angular.IPromise<Transaction | undefined> {
        const NOT_FOUND = 404;

        return this.$http
            .get(`${this.path(securityId)}/transactions/last`, {
                params: {
                    account_type: accountType,
                },
            })
            .then(
                (response: angular.IHttpResponse<Transaction>): Transaction =>
                    response.data,
            )
            .catch((error: unknown): undefined => {
                const { status, statusText, data } =
                    error as angular.IHttpResponse<string>;

                // Ignore if no last transaction
                if (NOT_FOUND === status) {
                    return;
                }

                throw new Error(`${status} ${statusText}: ${data}`);
            });
    }

    // Retrieves a single security
    public find(id: number): angular.IPromise<Security> {
        return this.$http
            .get(this.path(id), {
                cache: this.cache,
            })
            .then((response: angular.IHttpResponse<Security>): Security => {
                this.addRecent(response.data);

                return response.data;
            });
    }

    // Saves a security
    public save(security: Security): angular.IHttpPromise<Security> {
        // Flush the $http cache
        this.flush();

        return this.$http({
            method: undefined === security.id ? "POST" : "PATCH",
            url: this.path(security.id),
            data: security,
        });
    }

    // Deletes a security
    public destroy(security: Security): angular.IPromise<void> {
        // Flush the $http cache
        this.flush();

        return this.$http
            .delete(this.path(security.id))
            .then((): void => this.removeRecent(Number(security.id)));
    }

    // Favourites/unfavourites a security
    public toggleFavourite(security: Security): angular.IPromise<boolean> {
        // Flush the $http cache
        this.flush();

        return this.$http({
            method: security.favourite ? "DELETE" : "PUT",
            url: `${this.path(security.id)}/favourite`,
        }).then((): boolean => !security.favourite);
    }

    // Flush the cache
    public flush(id?: number): void {
        if (undefined === id) {
            this.cache.removeAll();
        } else {
            this.cache.remove(this.path(id));
        }
    }

    // Put an item into the LRU cache
    public addRecent(security: Security): void {
        // Put the item into the LRU cache
        this.recent = this.lruCache.put(security);

        // Update local storage with the new list
        this.$window.localStorage.setItem(
            this.LRU_LOCAL_STORAGE_KEY,
            JSON.stringify(this.lruCache.list),
        );
    }

    // Remove an item from the LRU cache
    public removeRecent(id: number): void {
        // Remove the item from the LRU cache
        this.recent = this.lruCache.remove(id);

        // Update local storage with the new list
        this.$window.localStorage.setItem(
            this.LRU_LOCAL_STORAGE_KEY,
            JSON.stringify(this.lruCache.list),
        );
    }
}

SecurityModel.$inject = [
    "$http",
    "$cacheFactory",
    "$window",
    "ogLruCacheFactory",
];