/*
 This file is part of GNU Taler
 (C) 2019 Taler Systems S.A.
 GNU Taler is free software; you can redistribute it and/or modify it under the
 terms of the GNU General Public License as published by the Free Software
 Foundation; either version 3, or (at your option) any later version.
 GNU Taler is distributed in the hope that it will be useful, but WITHOUT ANY
 WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR
 A PARTICULAR PURPOSE.  See the GNU General Public License for more details.
 You should have received a copy of the GNU General Public License along with
 GNU Taler; see the file COPYING.  If not, see 
 SPDX-License-Identifier: AGPL3.0-or-later
*/
/**
 * Imports.
 */
import {
  DEFAULT_REQUEST_TIMEOUT_MS,
  Headers,
  HttpRequestLibrary,
  HttpRequestOptions,
  HttpResponse,
} from "../util/http.js";
import { RequestThrottler } from "@gnu-taler/taler-util";
import axios, { AxiosResponse } from "axios";
import { TalerError } from "../errors.js";
import { Logger, bytesToString } from "@gnu-taler/taler-util";
import { TalerErrorCode, URL } from "@gnu-taler/taler-util";
const logger = new Logger("NodeHttpLib.ts");
/**
 * Implementation of the HTTP request library interface for node.
 */
export class NodeHttpLib implements HttpRequestLibrary {
  private throttle = new RequestThrottler();
  private throttlingEnabled = true;
  /**
   * Set whether requests should be throttled.
   */
  setThrottling(enabled: boolean): void {
    this.throttlingEnabled = enabled;
  }
  async fetch(url: string, opt?: HttpRequestOptions): Promise {
    const method = opt?.method ?? "GET";
    let body = opt?.body;
    logger.trace(`Requesting ${method} ${url}`);
    const parsedUrl = new URL(url);
    if (this.throttlingEnabled && this.throttle.applyThrottle(url)) {
      throw TalerError.fromDetail(
        TalerErrorCode.WALLET_HTTP_REQUEST_THROTTLED,
        {
          requestMethod: method,
          requestUrl: url,
          throttleStats: this.throttle.getThrottleStats(url),
        },
        `request to origin ${parsedUrl.origin} was throttled`,
      );
    }
    let timeoutMs: number | undefined;
    if (typeof opt?.timeout?.d_ms === "number") {
      timeoutMs = opt.timeout.d_ms;
    } else {
      timeoutMs = DEFAULT_REQUEST_TIMEOUT_MS;
    }
    // FIXME: Use AbortController / etc. to handle cancellation
    let resp: AxiosResponse;
    try {
      let respPromise = axios.default({
        method,
        url: url,
        responseType: "arraybuffer",
        headers: opt?.headers,
        validateStatus: () => true,
        transformResponse: (x) => x,
        data: body,
        timeout: timeoutMs,
        maxRedirects: 0,
      });
      if (opt?.cancellationToken) {
        respPromise = opt.cancellationToken.racePromise(respPromise);
      }
      resp = await respPromise;
    } catch (e: any) {
      throw TalerError.fromDetail(
        TalerErrorCode.WALLET_NETWORK_ERROR,
        {
          requestUrl: url,
          requestMethod: method,
        },
        `${e.message}`,
      );
    }
    const makeText = async (): Promise => {
      opt?.cancellationToken?.throwIfCancelled();
      const respText = new Uint8Array(resp.data);
      return bytesToString(respText);
    };
    const makeJson = async (): Promise => {
      opt?.cancellationToken?.throwIfCancelled();
      let responseJson;
      const respText = await makeText();
      try {
        responseJson = JSON.parse(respText);
      } catch (e) {
        logger.trace(`invalid json: '${resp.data}'`);
        throw TalerError.fromDetail(
          TalerErrorCode.WALLET_RECEIVED_MALFORMED_RESPONSE,
          {
            httpStatusCode: resp.status,
            requestUrl: url,
            requestMethod: method,
          },
          "Could not parse response body as JSON",
        );
      }
      if (responseJson === null || typeof responseJson !== "object") {
        logger.trace(`invalid json (not an object): '${respText}'`);
        throw TalerError.fromDetail(
          TalerErrorCode.WALLET_RECEIVED_MALFORMED_RESPONSE,
          {
            httpStatusCode: resp.status,
            requestUrl: url,
            requestMethod: method,
          },
          `invalid JSON`,
        );
      }
      return responseJson;
    };
    const makeBytes = async () => {
      opt?.cancellationToken?.throwIfCancelled();
      if (typeof resp.data.byteLength !== "number") {
        throw Error("expected array buffer");
      }
      const buf = resp.data;
      return buf;
    };
    const headers = new Headers();
    for (const hn of Object.keys(resp.headers)) {
      headers.set(hn, resp.headers[hn]);
    }
    return {
      requestUrl: url,
      requestMethod: method,
      headers,
      status: resp.status,
      text: makeText,
      json: makeJson,
      bytes: makeBytes,
    };
  }
  async get(url: string, opt?: HttpRequestOptions): Promise {
    return this.fetch(url, {
      method: "GET",
      ...opt,
    });
  }
  async postJson(
    url: string,
    body: any,
    opt?: HttpRequestOptions,
  ): Promise {
    return this.fetch(url, {
      method: "POST",
      body,
      ...opt,
    });
  }
}