import { AssertionFailed } from '../../errors';
import { error, Result, value } from 'defekt';

const assertFunctionIsNotThrowingAsync = async function <TError extends Error = Error> (
  actual: (...args: any[]) => any,
  expected?: string | RegExp | ((ex: TError) => boolean)
): Promise<Result<undefined, AssertionFailed>> {
  try {
    const promise = actual();

    if (!(promise instanceof Promise)) {
      return error(new AssertionFailed({
        message: 'The function did not return a Promise.'
      }));
    }

    await promise;
  // eslint-disable-next-line @typescript-eslint/no-implicit-any-catch
  } catch (ex: any) {
    if (expected === undefined) {
      return error(new AssertionFailed({
        message: 'The function threw an unexpected asynchronous exception.',
        actual: `Error message:\n${ex.message}`
      }));
    }

    if (expected instanceof RegExp && expected.test(ex.message)) {
      return error(new AssertionFailed({
        message: 'The function threw an unexpected asynchronous exception.',
        expected: `The message should not have matched:\n${expected.toString()}`,
        actual: `Error message:\n${ex.message}`
      }));
    }
    if (typeof expected === 'function' && expected(ex)) {
      return error(new AssertionFailed({
        message: 'The function threw an unexpected asynchronous exception.',
        expected: `The exception should not have fulfilled a predicate.`,
        actual: `Error message:\n${ex.message}`
      }));
    }

    if (typeof expected === 'string' && ex.message === expected) {
      return error(new AssertionFailed({
        message: 'The function threw an unexpected asynchronous exception.',
        expected: `The message should not have been:\n${expected}`,
        actual: `Error message:\n${ex.message}`
      }));
    }
  }

  return value();
};

export {
  assertFunctionIsNotThrowingAsync
};
