import * as path from 'path';
import * as assert from 'assert';
import { Request, Response } from 'express';

import TestServer from './lib/TestServer';
import eventually from './eventually';
import { Builder, Lanthan } from 'lanthan';
import { WebDriver } from 'selenium-webdriver';
import Page from './lib/Page';

describe('buffer command test', () => {
  const server = new TestServer().handle('/*', (req: Request, res: Response) => {
    res.send(`
      <!DOCTYPE html>
      <html lang="en">
        <head>
          <title>my_${req.path.slice(1)}</title>
        </head>
      </html>`);
  });
  let lanthan: Lanthan;
  let webdriver: WebDriver;
  let browser: any;

  before(async() => {
    lanthan = await Builder
      .forBrowser('firefox')
      .spyAddon(path.join(__dirname, '..'))
      .build();
    webdriver = lanthan.getWebDriver();
    browser = lanthan.getWebExtBrowser();
    await server.start();
  });

  after(async() => {
    await server.stop();
    if (lanthan) {
      await lanthan.quit();
    }
  });

  beforeEach(async() => {
    const tabs = await browser.tabs.query({});
    for (const tab of tabs.slice(1)) {
      await browser.tabs.remove(tab.id);
    }
    await browser.tabs.update(tabs[0].id, { url: server.url('/site1') });
    for (let i = 2; i <= 5; ++i) {
      await browser.tabs.create({ url: server.url('/site' + i) });
    }

    await eventually(async() => {
      const handles = await webdriver.getAllWindowHandles();
      assert.strictEqual(handles.length, 5);
      await webdriver.switchTo().window(handles[2]);
    });
  });

  it('should do nothing by buffer command with no parameters', async() => {
    const page = await Page.currentContext(webdriver);
    const console = await page.showConsole();
    await console.execCommand('buffer');

    await eventually(async() => {
      const tabs = await browser.tabs.query({ active: true });
      assert.strictEqual(tabs[0].index, 2);
    });
  });

  it('should select a tab by buffer command with a number', async() => {
    const page = await Page.currentContext(webdriver);
    const console = await page.showConsole();
    await console.execCommand('buffer 1');

    await eventually(async() => {
      const tabs = await browser.tabs.query({ active: true });
      assert.strictEqual(tabs[0].index, 0);
    });
  });

  it('should should an out of range error by buffer commands', async() => {
    const page = await Page.currentContext(webdriver);
    let console = await page.showConsole();
    await console.execCommand('buffer 0');

    await eventually(async() => {
      const text = await console.getErrorMessage();
      assert.strictEqual(text, 'tab 0 does not exist');
    });

    await (webdriver.switchTo() as any).parentFrame();

    console = await page.showConsole();
    await console.execCommand('buffer 9');

    await eventually(async() => {
      const text = await console.getErrorMessage();
      assert.strictEqual(text, 'tab 9 does not exist');
    });
  });

  it('should select a tab by buffer command with a title', async() => {
    const page = await Page.currentContext(webdriver);
    const console = await page.showConsole();
    await console.execCommand('buffer my_site1');

    await eventually(async() => {
      const tabs = await browser.tabs.query({ active: true });
      assert.strictEqual(tabs[0].index, 0);
    });
  });

  it('should select a tab by buffer command with an URL', async() => {
    const page = await Page.currentContext(webdriver);
    const console = await page.showConsole();
    await console.execCommand('buffer /site1');

    await eventually(async() => {
      const tabs = await browser.tabs.query({ active: true });
      assert.strictEqual(tabs[0].index, 0);
    });
  });

  it('should select tabs rotately', async() => {
    const handles = await webdriver.getAllWindowHandles();
    await webdriver.switchTo().window(handles[4]);

    const page = await Page.currentContext(webdriver);
    const console = await page.showConsole();
    await console.execCommand('buffer site');

    await eventually(async() => {
      const tabs = await browser.tabs.query({ active: true });
      assert.strictEqual(tabs[0].index, 0);
    });
  });

  it('should do nothing by ":buffer %"', async() => {
    const page = await Page.currentContext(webdriver);
    const console = await page.showConsole();
    await console.execCommand('buffer %');

    await eventually(async() => {
      const tabs = await browser.tabs.query({ active: true });
      assert.strictEqual(tabs[0].index, 2);
    });
  });

  it('should selects last selected tab by ":buffer #"', async() => {
    const handles = await webdriver.getAllWindowHandles();
    await webdriver.switchTo().window(handles[1]);

    const page = await Page.currentContext(webdriver);
    const console = await page.showConsole();
    await console.execCommand('buffer #');

    await eventually(async() => {
      const tabs = await browser.tabs.query({ active: true });
      assert.strictEqual(tabs[0].index, 2);
    });
  });
});