import { suite } from 'uvu';
import * as assert from 'uvu/assert';
import { createRuntime } from '../lib/runtime.js';
import { loadConfig } from '../lib/config.js';
import { promises as fsPromises } from 'fs';
import { relative as pathRelative } from 'path';
import { doc } from './test-utils.js';

const { readdir, stat } = fsPromises;

const SnowpackDev = suite('snowpack.dev');

let runtime, cwd, setupError;

SnowpackDev.before(async () => {
  // Bug: Snowpack config is still loaded relative to the current working directory.
  cwd = process.cwd();
  process.chdir(new URL('../examples/snowpack/', import.meta.url).pathname);

  const astroConfig = await loadConfig(new URL('../examples/snowpack', import.meta.url).pathname);

  const logging = {
    level: 'error',
    dest: process.stderr,
  };

  try {
    runtime = await createRuntime(astroConfig, logging);
  } catch (err) {
    console.error(err);
    setupError = err;
  }
});

SnowpackDev.after(async () => {
  process.chdir(cwd);
  (await runtime) && runtime.shutdown();
});

async function* allPageFiles(root) {
  for (const filename of await readdir(root)) {
    const fullpath = new URL(filename, root);
    const info = await stat(fullpath);

    if (info.isDirectory()) {
      yield* allPageFiles(new URL(fullpath + '/'));
    } else {
      yield fullpath;
    }
  }
}

async function* allPages(root) {
  for await (let fileURL of allPageFiles(root)) {
    let bare = fileURL.pathname.replace(/\.(astro|md)$/, '').replace(/index$/, '');

    yield '/' + pathRelative(root.pathname, bare);
  }
}

SnowpackDev('No error creating the runtime', () => {
  assert.equal(setupError, undefined);
});

SnowpackDev('Can load every page', async () => {
  const failed = [];

  const pageRoot = new URL('../examples/snowpack/astro/pages/', import.meta.url);
  for await (let pathname of allPages(pageRoot)) {
    if (pathname.includes('proof-of-concept-dynamic')) {
      continue;
    }
    const result = await runtime.load(pathname);
    if (result.statusCode === 500) {
      failed.push({ ...result, pathname });
      continue;
    }
    assert.equal(result.statusCode, 200, `Loading ${pathname}`);
  }

  if (failed.length > 0) {
    console.error(failed);
  }
  assert.equal(failed.length, 0, 'Failed pages');
});

SnowpackDev.run();