/**
 * Copyright Microsoft Corporation. All rights reserved.
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *     http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */
import * as fs from 'fs';
import { test, expect, playwrightCtConfigText, stripAnsi } from './playwright-test-fixtures';
import { execSync } from 'child_process';
test.describe.configure({ mode: 'parallel' });
function trimPatch(patch: string) {
  return patch.split('\n').map(line => line.trimEnd()).join('\n');
}
test('should update snapshot with the update-snapshots flag with multiple projects', async ({ runInlineTest }, testInfo) => {
  const result = await runInlineTest({
    '.git/marker': '',
    'playwright.config.ts': `
      export default { projects: [{ name: 'p1' }, { name: 'p2' }] };
    `,
    'a.spec.ts': `
      import { test, expect } from '@playwright/test';
      test('test', async ({ page }) => {
        await page.setContent(\`
hello bye \`);
        await expect(page.locator('body')).toMatchAriaSnapshot(\`
          - heading "world"
        \`);
      });
    `
  }, { 'update-snapshots': true });
  expect(result.exitCode).toBe(0);
  const patchPath = testInfo.outputPath('test-results/rebaselines.patch');
  const data = fs.readFileSync(patchPath, 'utf-8');
  expect(trimPatch(data)).toBe(`diff --git a/a.spec.ts b/a.spec.ts
--- a/a.spec.ts
+++ b/a.spec.ts
@@ -3,7 +3,8 @@
       test('test', async ({ page }) => {
         await page.setContent(\`hello bye \`);
         await expect(page.locator('body')).toMatchAriaSnapshot(\`
-          - heading "world"
+          - heading "hello" [level=1]
+          - heading "bye" [level=2]
         \`);
       });
\\ No newline at end of file
`);
  expect(stripAnsi(result.output).replace(/\\/g, '/')).toContain(`New baselines created for:
  a.spec.ts
  git apply test-results/rebaselines.patch
`);
  execSync(`patch -p1 < ${patchPath}`, { cwd: testInfo.outputPath() });
  const result2 = await runInlineTest({});
  expect(result2.exitCode).toBe(0);
});
test('should update missing snapshots', async ({ runInlineTest }, testInfo) => {
  const result = await runInlineTest({
    '.git/marker': '',
    'a.spec.ts': `
      import { test, expect } from '@playwright/test';
      test('test', async ({ page }) => {
        await page.setContent(\`hello \`);
        await expect(page.locator('body')).toMatchAriaSnapshot(\`\`);
      });
    `
  });
  expect(result.exitCode).toBe(0);
  expect(stripAnsi(result.output).replace(/\\/g, '/')).toContain(`New baselines created for:
  a.spec.ts
  git apply test-results/rebaselines.patch
`);
  const patchPath = testInfo.outputPath('test-results/rebaselines.patch');
  const data = fs.readFileSync(patchPath, 'utf-8');
  expect(trimPatch(data)).toBe(`diff --git a/a.spec.ts b/a.spec.ts
--- a/a.spec.ts
+++ b/a.spec.ts
@@ -2,6 +2,8 @@
       import { test, expect } from '@playwright/test';
       test('test', async ({ page }) => {
         await page.setContent(\`hello \`);
-        await expect(page.locator('body')).toMatchAriaSnapshot(\`\`);
+        await expect(page.locator('body')).toMatchAriaSnapshot(\`
+          - heading "hello" [level=1]
+        \`);
       });
\\ No newline at end of file
`);
  execSync(`patch -p1 < ${patchPath}`, { cwd: testInfo.outputPath() });
  const result2 = await runInlineTest({});
  expect(result2.exitCode).toBe(0);
});
test('should update multiple missing snapshots', async ({ runInlineTest }, testInfo) => {
  const result = await runInlineTest({
    '.git/marker': '',
    'a.spec.ts': `
      import { test, expect } from '@playwright/test';
      test('test', async ({ page }) => {
        await page.setContent(\`hello \`);
        await expect(page.locator('body')).toMatchAriaSnapshot(\`\`);
        await expect(page.locator('body')).toMatchAriaSnapshot(\`\`);
      });
    `
  });
  expect(result.exitCode).toBe(0);
  expect(stripAnsi(result.output).replace(/\\/g, '/')).toContain(`New baselines created for:
  a.spec.ts
  git apply test-results/rebaselines.patch
`);
  const patchPath = testInfo.outputPath('test-results/rebaselines.patch');
  const data = fs.readFileSync(patchPath, 'utf-8');
  expect(trimPatch(data)).toBe(`diff --git a/a.spec.ts b/a.spec.ts
--- a/a.spec.ts
+++ b/a.spec.ts
@@ -2,7 +2,11 @@
       import { test, expect } from '@playwright/test';
       test('test', async ({ page }) => {
         await page.setContent(\`hello \`);
-        await expect(page.locator('body')).toMatchAriaSnapshot(\`\`);
-        await expect(page.locator('body')).toMatchAriaSnapshot(\`\`);
+        await expect(page.locator('body')).toMatchAriaSnapshot(\`
+          - heading "hello" [level=1]
+        \`);
+        await expect(page.locator('body')).toMatchAriaSnapshot(\`
+          - heading "hello" [level=1]
+        \`);
       });
\\ No newline at end of file
`);
  execSync(`patch -p1 < ${patchPath}`, { cwd: testInfo.outputPath() });
  const result2 = await runInlineTest({});
  expect(result2.exitCode).toBe(0);
});
test('should generate baseline with regex', async ({ runInlineTest }, testInfo) => {
  const result = await runInlineTest({
    '.git/marker': '',
    'a.spec.ts': `
      import { test, expect } from '@playwright/test';
      test('test', async ({ page }) => {
        await page.setContent(\`
          Item 1 
          Item 2 
          Time 15:30 
          Year 2022 
          Duration 12ms 
          22,333 
          2,333.79 
          Total 22 
          /Regex 1/ 
          /Regex 22ms/ 
         \`);
        await expect(page.locator('body')).toMatchAriaSnapshot(\`\`);
      });
    `
  });
  expect(result.exitCode).toBe(0);
  const patchPath = testInfo.outputPath('test-results/rebaselines.patch');
  const data = fs.readFileSync(patchPath, 'utf-8');
  expect(trimPatch(data)).toBe(`diff --git a/a.spec.ts b/a.spec.ts
--- a/a.spec.ts
+++ b/a.spec.ts
@@ -13,6 +13,18 @@
           /Regex 1/ 
           /Regex 22ms/ 
         \`);
-        await expect(page.locator('body')).toMatchAriaSnapshot(\`\`);
+        await expect(page.locator('body')).toMatchAriaSnapshot(\`
+          - list:
+            - listitem: Item 1
+            - listitem: Item 2
+            - listitem: /Time \\\\d+:\\\\d+/
+            - listitem: /Year \\\\d+/
+            - listitem: /Duration \\\\d+[hmsp]+/
+            - listitem: /\\\\d+,\\\\d+/
+            - listitem: /\\\\d+,\\\\d+\\\\.\\\\d+/
+            - listitem: /Total \\\\d+/
+            - listitem: /Regex 1/
+            - listitem: /\\\\/Regex \\\\d+[hmsp]+\\\\//
+        \`);
       });
\\ No newline at end of file
`);
  execSync(`patch -p1 < ${patchPath}`, { cwd: testInfo.outputPath() });
  const result2 = await runInlineTest({});
  expect(result2.exitCode).toBe(0);
});
test('should generate baseline with special characters', async ({ runInlineTest }, testInfo) => {
  const result = await runInlineTest({
    '.git/marker': '',
    'a.spec.ts': `
      import { test, expect } from '@playwright/test';
      test('test', async ({ page }) => {
        await page.setContent(\`
          
            one: link1  "two link2  'three link3  \\\`four 
           
          heading "name" [level=1] 
          Click: me 
          Click: 123 
          Click ' me 
          Click: ' me 
          Click " me 
          Click " me 123 
          Click \\\\ me 
          Click \\\\ me 123 
          Item: 1 
          Item {a: b} 
         \`);
        await expect(page.locator('body')).toMatchAriaSnapshot(\`\`);
      });
    `
  });
  expect(result.exitCode).toBe(0);
  const patchPath = testInfo.outputPath('test-results/rebaselines.patch');
  const data = fs.readFileSync(patchPath, 'utf-8');
  expect(trimPatch(data)).toBe(`diff --git a/a.spec.ts b/a.spec.ts
--- a/a.spec.ts
+++ b/a.spec.ts
@@ -17,6 +17,27 @@
           Item: 1 
           Item {a: b} 
         \`);
-        await expect(page.locator('body')).toMatchAriaSnapshot(\`\`);
+        await expect(page.locator('body')).toMatchAriaSnapshot(\`
+          - list:
+            - group:
+              - text: "one:"
+              - link "link1"
+              - text: "\\\\\"two"
+              - link "link2"
+              - text: "'three"
+              - link "link3"
+              - text: "\\\`four"
+            - heading "heading \\\\"name\\\\" [level=1]" [level=1]
+            - 'button "Click: me"'
+            - 'button /Click: \\\\d+/'
+            - button "Click ' me"
+            - 'button "Click: '' me"'
+            - button "Click \\\\" me"
+            - button /Click " me \\\\d+/
+            - button "Click \\\\\\\\ me"
+            - button /Click \\\\\\\\ me \\\\d+/
+            - listitem: \"Item: 1\"
+            - listitem: \"Item {a: b}\"
+        \`);
       });
\\ No newline at end of file
`);
  execSync(`patch -p1 < ${patchPath}`, { cwd: testInfo.outputPath() });
  const result2 = await runInlineTest({});
  expect(result2.exitCode).toBe(0);
});
test('should update missing snapshots in tsx', async ({ runInlineTest }, testInfo) => {
  const result = await runInlineTest({
    '.git/marker': '',
    'playwright.config.ts': playwrightCtConfigText,
    'playwright/index.html': ``,
    'playwright/index.ts': ``,
    'src/button.tsx': `
      export const Button = () => Button ;
    `,
    'src/button.test.tsx': `
      import { test, expect } from '@playwright/experimental-ct-react';
      import { Button } from './button.tsx';
      test('pass', async ({ mount }) => {
        const component = await mount(Button ;
    `,
    'src/button-1.test.tsx': `
      import { test, expect } from '@playwright/experimental-ct-react';
      import { Button } from './button.tsx';
      test('pass 1', async ({ mount }) => {
        const component = await mount(hello ');
        await expect(page.locator('div')).toMatchAriaSnapshot('- heading', { timeout: 3000 });
      });
    `,
  }, { 'update-snapshots': true });
  expect(result.exitCode).toBe(1);
  const patchPath = testInfo.outputPath('test-results/rebaselines.patch');
  expect(fs.existsSync(patchPath)).toBe(false);
  expect(result.output).not.toContain('New baselines created');
  expect(result.output).toContain('Expected: "- heading"');
  expect(result.output).toContain('Received: ');
});
test.describe('update-snapshots none', () => {
  test('should create new baseline for matching snapshot', async ({ runInlineTest }, testInfo) => {
    const result = await runInlineTest({
      '.git/marker': '',
      'a.spec.ts': `
        import { test, expect } from '@playwright/test';
        test('test', async ({ page }) => {
          await page.setContent(\`hello world \`);
          await expect(page.locator('body')).toMatchAriaSnapshot(\`\`);
        });
      `
    }, { 'update-snapshots': 'none' });
    expect(result.exitCode).toBe(1);
    const patchPath = testInfo.outputPath('test-results/rebaselines.patch');
    expect(fs.existsSync(patchPath)).toBeFalsy();
  });
});
test.describe('update-snapshots all', () => {
  test('should create new baseline for matching snapshot', async ({ runInlineTest }, testInfo) => {
    const result = await runInlineTest({
      '.git/marker': '',
      'a.spec.ts': `
        import { test, expect } from '@playwright/test';
        test('test', async ({ page }) => {
          await page.setContent(\`hello world \`);
          await expect(page.locator('body')).toMatchAriaSnapshot(\`
            - heading "hello"
          \`);
        });
      `
    }, { 'update-snapshots': 'all' });
    expect(result.exitCode).toBe(0);
    const patchPath = testInfo.outputPath('test-results/rebaselines.patch');
    const data = fs.readFileSync(patchPath, 'utf-8');
    expect(trimPatch(data)).toBe(`diff --git a/a.spec.ts b/a.spec.ts
--- a/a.spec.ts
+++ b/a.spec.ts
@@ -3,7 +3,8 @@
         test('test', async ({ page }) => {
           await page.setContent(\`hello world \`);
           await expect(page.locator('body')).toMatchAriaSnapshot(\`
-            - heading "hello"
+            - heading "hello" [level=1]
+            - heading "world" [level=1]
           \`);
         });
\\ No newline at end of file
`);
    expect(stripAnsi(result.output).replace(/\\/g, '/')).toContain(`New baselines created for:
  a.spec.ts
  git apply test-results/rebaselines.patch
`);
    execSync(`patch -p1 < ${patchPath}`, { cwd: testInfo.outputPath() });
    const result2 = await runInlineTest({});
    expect(result2.exitCode).toBe(0);
  });
});
test.describe('update-source-method', () => {
  test('should overwrite source', async ({ runInlineTest }, testInfo) => {
    const result = await runInlineTest({
      '.git/marker': '',
      'a.spec.ts': `
        import { test, expect } from '@playwright/test';
        test('test', async ({ page }) => {
          await page.setContent(\`hello \`);
          await expect(page.locator('body')).toMatchAriaSnapshot(\`
            - heading "world"
          \`);
        });
      `
    }, { 'update-snapshots': 'all', 'update-source-method': 'overwrite' });
    expect(result.exitCode).toBe(0);
    const patchPath = testInfo.outputPath('test-results/rebaselines.patch');
    expect(fs.existsSync(patchPath)).toBeFalsy();
    const data = fs.readFileSync(testInfo.outputPath('a.spec.ts'), 'utf-8');
    expect(data).toBe(`
        import { test, expect } from '@playwright/test';
        test('test', async ({ page }) => {
          await page.setContent(\`hello \`);
          await expect(page.locator('body')).toMatchAriaSnapshot(\`
            - heading "hello" [level=1]
          \`);
        });
      `);
    expect(stripAnsi(result.output).replace(/\\/g, '/')).toContain(`New baselines created for:
  a.spec.ts
`);
    const result2 = await runInlineTest({});
    expect(result2.exitCode).toBe(0);
  });
  test('should 3way source', async ({ runInlineTest }, testInfo) => {
    const result = await runInlineTest({
      '.git/marker': '',
      'a.spec.ts': `
        import { test, expect } from '@playwright/test';
        test('test', async ({ page }) => {
          await page.setContent(\`hello \`);
          await expect(page.locator('body')).toMatchAriaSnapshot(\`
            - heading "world"
          \`);
        });
      `
    }, { 'update-snapshots': 'all', 'update-source-method': '3way' });
    expect(result.exitCode).toBe(0);
    const patchPath = testInfo.outputPath('test-results/rebaselines.patch');
    expect(fs.existsSync(patchPath)).toBeFalsy();
    const data = fs.readFileSync(testInfo.outputPath('a.spec.ts'), 'utf-8');
    expect(data).toBe(`
        import { test, expect } from '@playwright/test';
        test('test', async ({ page }) => {
          await page.setContent(\`hello \`);
          await expect(page.locator('body')).toMatchAriaSnapshot(\`
\<<<<<<< HEAD
            - heading "world"
=======
            - heading "hello" [level=1]
>>>>>>> SNAPSHOT
          \`);
        });
      `);
    expect(stripAnsi(result.output).replace(/\\/g, '/')).toContain(`New baselines created for:
  a.spec.ts
`);
  });
});