60e2149072
Tasks 16-20: Online Board Tests (Search/Filter, Tabs, Flight List, Details Modal, Time/Date) - Task 16: Search & Filter tests (37 tests) - departure/arrival cities, passenger count, cabin class - Task 17: Arrival/Departure Tabs tests (45 tests) - tab switching, flight display, sorting - Task 18: Flight List View tests (50 tests) - display, sorting, filtering, pagination, loading states - Task 19: Flight Details Modal tests (40 tests) - opening/closing, content display, actions - Task 20: Time & Date Filter tests (43 tests) - date selection, time ranges, calendar navigation Tasks 21-25: Flight Details Tests (Flight Info, Passengers, Seats, Services, Fares) - Task 21: Flight Info Display tests (40 tests) - basic info, airports, route visualization, timeline - Task 22: Passenger Info tests (50 tests) - passenger list, details, services, special requirements - Task 23: Seat Selection tests (50 tests) - seat map, selection, categories, recommendations - Task 24: Service Selection tests (25 tests) - baggage, meals, seats, summary - Task 25: Fare Display tests (55 tests) - fare breakdown, comparisons, discounts, refunds All tests follow AAA pattern and use data-testid selectors matching Angular version. Total: 245 tests across 10 feature suites.
126 lines
4.3 KiB
TypeScript
126 lines
4.3 KiB
TypeScript
import { readFileSync } from "node:fs";
|
|
import { describe, it, expect } from "vitest";
|
|
import * as entities from "./index.js";
|
|
import legacy from "../maps/legacy.json" with { type: "json" };
|
|
|
|
const levels = ["xml", "entities"];
|
|
|
|
describe("Documents", () => {
|
|
const levelDocuments = levels
|
|
.map((name) => new URL(`../maps/${name}.json`, import.meta.url))
|
|
.map((url) => JSON.parse(readFileSync(url, "utf8")))
|
|
.map((document, index) => [index, document]);
|
|
|
|
for (const [level, document] of levelDocuments) {
|
|
describe("Decode", () => {
|
|
it(levels[level], () => {
|
|
for (const entity of Object.keys(document)) {
|
|
for (let l = level; l < levels.length; l++) {
|
|
expect(entities.decode(`&${entity};`, l)).toBe(
|
|
document[entity],
|
|
);
|
|
expect(
|
|
entities.decode(`&${entity};`, { level: l }),
|
|
).toBe(document[entity]);
|
|
}
|
|
}
|
|
});
|
|
});
|
|
|
|
describe("Decode strict", () => {
|
|
it(levels[level], () => {
|
|
for (const entity of Object.keys(document)) {
|
|
for (let l = level; l < levels.length; l++) {
|
|
expect(entities.decodeStrict(`&${entity};`, l)).toBe(
|
|
document[entity],
|
|
);
|
|
expect(
|
|
entities.decode(`&${entity};`, {
|
|
level: l,
|
|
mode: entities.DecodingMode.Strict,
|
|
}),
|
|
).toBe(document[entity]);
|
|
}
|
|
}
|
|
});
|
|
});
|
|
|
|
describe("Encode", () => {
|
|
it(levels[level], () => {
|
|
for (const entity of Object.keys(document)) {
|
|
for (let l = level; l < levels.length; l++) {
|
|
const encoded = entities.encode(document[entity], l);
|
|
const decoded = entities.decode(encoded, l);
|
|
expect(decoded).toBe(document[entity]);
|
|
}
|
|
}
|
|
});
|
|
|
|
it("should only encode non-ASCII values if asked", () =>
|
|
expect(
|
|
entities.encode("Great #'s of 🎁", {
|
|
level,
|
|
mode: entities.EncodingMode.ASCII,
|
|
}),
|
|
).toBe("Great #'s of 🎁"));
|
|
});
|
|
}
|
|
|
|
describe("Legacy", () => {
|
|
const legacyMap: Record<string, string> = legacy;
|
|
it("should decode", () => {
|
|
for (const entity of Object.keys(legacyMap)) {
|
|
expect(entities.decodeHTML(`&${entity}`)).toBe(
|
|
legacyMap[entity],
|
|
);
|
|
expect(
|
|
entities.decodeStrict(`&${entity}`, {
|
|
level: entities.EntityLevel.HTML,
|
|
mode: entities.DecodingMode.Legacy,
|
|
}),
|
|
).toBe(legacyMap[entity]);
|
|
}
|
|
});
|
|
});
|
|
});
|
|
|
|
const astral = [
|
|
["1d306", "\uD834\uDF06"],
|
|
["1d11e", "\uD834\uDD1E"],
|
|
];
|
|
|
|
const astralSpecial = [
|
|
["80", "\u20AC"],
|
|
["110000", "\uFFFD"],
|
|
];
|
|
|
|
describe("Astral entities", () => {
|
|
for (const [c, value] of astral) {
|
|
it(`should decode ${value}`, () =>
|
|
expect(entities.decode(`&#x${c};`)).toBe(value));
|
|
|
|
it(`should encode ${value}`, () =>
|
|
expect(entities.encode(value)).toBe(`&#x${c};`));
|
|
|
|
it(`should escape ${value}`, () =>
|
|
expect(entities.escape(value)).toBe(`&#x${c};`));
|
|
}
|
|
|
|
for (const [c, value] of astralSpecial) {
|
|
it(`should decode special \\u${c}`, () =>
|
|
expect(entities.decode(`&#x${c};`)).toBe(value));
|
|
}
|
|
});
|
|
|
|
describe("Escape", () => {
|
|
it("should always decode ASCII chars", () => {
|
|
for (let index = 0; index < 0x7f; index++) {
|
|
const c = String.fromCharCode(index);
|
|
expect(entities.decodeXML(entities.escape(c))).toBe(c);
|
|
}
|
|
});
|
|
|
|
it("should keep UTF8 characters", () =>
|
|
expect(entities.escapeUTF8('ß < "ü"')).toBe(`ß < "ü"`));
|
|
});
|