├── .eslintrc.json
├── .github
└── workflows
│ └── lint-test.yml
├── .gitignore
├── README.md
├── example.js
├── index.js
├── lib
├── findTicket.js
├── parse.js
├── queryTicket.js
└── requestBackend.js
├── package-lock.json
├── package.json
└── test
├── index.js
├── ticket-456800002504-2023-07-10-erika-musterfrau.json
└── ticket-XXXXXX-2023-05-12-erika-musterman.json
/.eslintrc.json:
--------------------------------------------------------------------------------
1 | {
2 | "env": {
3 | "browser": true,
4 | "es2021": true
5 | },
6 | "extends": "eslint:recommended",
7 | "parserOptions": {
8 | "ecmaVersion": "latest",
9 | "sourceType": "module"
10 | },
11 | "rules": {
12 | }
13 | }
14 |
--------------------------------------------------------------------------------
/.github/workflows/lint-test.yml:
--------------------------------------------------------------------------------
1 | name: lint & test
2 |
3 | on:
4 | push:
5 | branches:
6 | - '*'
7 | pull_request:
8 | branches:
9 | - '*'
10 |
11 | jobs:
12 | lint-test:
13 | runs-on: ubuntu-latest
14 | strategy:
15 | matrix:
16 | node-version: ['16', '18', '20']
17 |
18 | steps:
19 | - name: checkout
20 | uses: actions/checkout@v4
21 | - name: setup Node v${{ matrix.node-version }}
22 | uses: actions/setup-node@v3
23 | with:
24 | node-version: ${{ matrix.node-version }}
25 | - run: npm install
26 |
27 | - run: npm run lint
28 | - run: npm test
29 |
--------------------------------------------------------------------------------
/.gitignore:
--------------------------------------------------------------------------------
1 | .DS_Store
2 | # Logs
3 | logs
4 | *.log
5 | npm-debug.log*
6 | yarn-debug.log*
7 | yarn-error.log*
8 | lerna-debug.log*
9 |
10 | # Diagnostic reports (https://nodejs.org/api/report.html)
11 | report.[0-9]*.[0-9]*.[0-9]*.[0-9]*.json
12 |
13 | # Runtime data
14 | pids
15 | *.pid
16 | *.seed
17 | *.pid.lock
18 |
19 | # Directory for instrumented libs generated by jscoverage/JSCover
20 | lib-cov
21 |
22 | # Coverage directory used by tools like istanbul
23 | coverage
24 | *.lcov
25 |
26 | # nyc test coverage
27 | .nyc_output
28 |
29 | # Grunt intermediate storage (https://gruntjs.com/creating-plugins#storing-task-files)
30 | .grunt
31 |
32 | # Bower dependency directory (https://bower.io/)
33 | bower_components
34 |
35 | # node-waf configuration
36 | .lock-wscript
37 |
38 | # Compiled binary addons (https://nodejs.org/api/addons.html)
39 | build/Release
40 |
41 | # Dependency directories
42 | node_modules/
43 | jspm_packages/
44 |
45 | # TypeScript v1 declaration files
46 | typings/
47 |
48 | # TypeScript cache
49 | *.tsbuildinfo
50 |
51 | # Optional npm cache directory
52 | .npm
53 |
54 | # Optional eslint cache
55 | .eslintcache
56 |
57 | # Microbundle cache
58 | .rpt2_cache/
59 | .rts2_cache_cjs/
60 | .rts2_cache_es/
61 | .rts2_cache_umd/
62 |
63 | # Optional REPL history
64 | .node_repl_history
65 |
66 | # Output of 'npm pack'
67 | *.tgz
68 |
69 | # Yarn Integrity file
70 | .yarn-integrity
71 |
72 | # dotenv environment variables file
73 | .env
74 | .env.test
75 |
76 | # parcel-bundler cache (https://parceljs.org/)
77 | .cache
78 |
79 | # Next.js build output
80 | .next
81 |
82 | # Nuxt.js build / generate output
83 | .nuxt
84 | dist
85 |
86 | # Gatsby files
87 | .cache/
88 | # Comment in the public line in if your project uses Gatsby and *not* Next.js
89 | # https://nextjs.org/blog/next-9-1#public-directory-support
90 | # public
91 |
92 | # vuepress build output
93 | .vuepress/dist
94 |
95 | # Serverless directories
96 | .serverless/
97 |
98 | # FuseBox cache
99 | .fusebox/
100 |
101 | # DynamoDB Local files
102 | .dynamodb/
103 |
104 | # TernJS port file
105 | .tern-port
106 |
107 | # Editor directories and files
108 | .idea
109 | .vscode
110 | *.suo
111 | *.ntvs*
112 | *.njsproj
113 | *.sln
114 | *.sw?
115 | *.sublime-project
116 | *.sublime-workspace
117 | yarn.lock
118 | jsconfig.json
119 |
--------------------------------------------------------------------------------
/README.md:
--------------------------------------------------------------------------------
1 | # db-tickets
2 | A library to retrieve ticket information from Deutsche Bahn by providing ticket number and last name.
3 |
4 | ## How to use
5 |
6 | ```sh
7 | npm install db-tickets
8 | ```
9 |
10 | ```js
11 | import queryTicket from 'db-tickets'
12 | queryTicket('W7KHTA', 'Mustermann').then((response) => console.log(response))
13 | ```
14 |
15 | ```json
16 | {
17 | "order": {
18 | "ticketNumber": "W7KHTA",
19 | "bookingDate": "2023-01-21T12:30:03",
20 | "validFrom": "2023-01-21T00:00:00",
21 | "validUntil": "2023-01-21T23:59:59",
22 | "journeyStart": "2023-01-21T12:40:00",
23 | "firstName": "Max",
24 | "lastName": "Mustermann",
25 | "text": "Einfache Fahrt, Regio120 Ticket, 2. Kl., 1 Erw., Jena/Leipzig#",
26 | "class": "S2"
27 | },
28 | "journey": {
29 | "type": "journey",
30 | "id": "51451984",
31 | "legs": [
32 | {
33 | "origin": {
34 | "type": "station",
35 | "id": "8011956",
36 | "name": "Jena Paradies",
37 | "location": {
38 | "type": "location",
39 | "longitude": 11.587464,
40 | "latitude": 50.924853
41 | }
42 | },
43 | "destination": {
44 | "type": "station",
45 | "id": "8010205",
46 | "name": "Leipzig Hbf",
47 | "location": {
48 | "type": "location",
49 | "longitude": 12.382066,
50 | "latitude": 51.345467
51 | }
52 | },
53 | "line": {
54 | "type": "line",
55 | "id": "C0-0.0",
56 | "name": "RE 4986",
57 | "mode": "train"
58 | },
59 | "departure": "2023-01-21T12:40:00+01:00",
60 | "departurePlatform": "2",
61 | "arrival": "2023-01-21T13:52:00+01:00",
62 | "arrivalPlatform": "9",
63 | "mode": "train",
64 | "public": true
65 | }
66 | ],
67 | "price": {
68 | "amount": "18.60",
69 | "currency": "EUR"
70 | }
71 | }
72 | }
73 | ```
74 |
--------------------------------------------------------------------------------
/example.js:
--------------------------------------------------------------------------------
1 | import queryTicket from './index.js'
2 | import util from 'util'
3 |
4 | // request data with ticket number and lastname
5 | const ticketNumber = process.env.TICKET_NR || 'XXXXXX'
6 | const lastname = process.env.LAST_NAME || 'Mustermann'
7 |
8 | const data = await queryTicket(ticketNumber, lastname)
9 | // console.log(util.inspect(data, false, null))
10 | console.log(JSON.stringify(data))
11 |
--------------------------------------------------------------------------------
/index.js:
--------------------------------------------------------------------------------
1 | /*!
2 | * db-tickets
3 | * ISC Licensed
4 | */
5 |
6 | import {queryTicket as rqorderdetails} from './lib/queryTicket.js'
7 | import findTicket from './lib/findTicket.js'
8 |
9 | /**
10 | * queryTicket function, returns the ticket data based on the ticket number and last name. handles both old and new ticket system
11 | * @param string | DB ticket number (ticket number up 12 digits or 6 characters)
12 | * @param lastName | last name of purchaser
13 | * @return Promise | fulfills with an object that contains the ticket data
14 | */
15 | const queryTicket = async (ticketNumber, lastName) => {
16 | //create a null kwid variable
17 | let kwid = null
18 | if (ticketNumber.length > 5) {
19 | const order = await findTicket(ticketNumber, lastName)
20 | //extract the kwid from the order object
21 | kwid = order.kwid
22 | }
23 | const ticketData = await rqorderdetails(ticketNumber, lastName, kwid)
24 | return ticketData
25 | }
26 |
27 |
28 | export default queryTicket
29 |
--------------------------------------------------------------------------------
/lib/findTicket.js:
--------------------------------------------------------------------------------
1 | import { requestBackend } from './requestBackend.js'
2 | /**
3 | * findTicket function
4 | * @param string | DB ticket number (Auftragsnummer 12 digits)
5 | * @param lastName | last name of purchaser
6 | * @return Promise | fulfills with an object that contains the ticket data
7 | */
8 | export const findTicket = async (ticketNumber, lastName) => {
9 | //make a request to the endpoint with the following body:
10 | //
11 | const reqBody = ''
12 | + ''
13 | + ''
14 | return requestBackend(reqBody).then(json => {
15 | const data = json.rporderheadlist
16 |
17 | const _order = data['orderheadlist'][0]['orderhead'][0]['$']
18 | return _order})
19 | }
20 |
21 |
22 | export default findTicket
--------------------------------------------------------------------------------
/lib/parse.js:
--------------------------------------------------------------------------------
1 | const modeCodes = {
2 | T: 'train'
3 | // todo: letters for other modes
4 | }
5 |
6 | const date = (_date) => {
7 | // merge date fields
8 | let date = new Date(_date['dt'].slice(0, 10))
9 | const t = _date['t'].split(':')
10 | date.setHours(t[0], t[1])
11 |
12 | // convert to extended ISO 8601 with timezone
13 | const tzOffset = -date.getTimezoneOffset()
14 | const diff = tzOffset >= 0 ? '+' : '-'
15 | const pad = n => `${Math.floor(Math.abs(n))}`.padStart(2, '0')
16 | return date.getFullYear() +
17 | '-' + pad(date.getMonth() + 1) +
18 | '-' + pad(date.getDate()) +
19 | 'T' + pad(date.getHours()) +
20 | ':' + pad(date.getMinutes()) +
21 | ':' + pad(date.getSeconds()) +
22 | diff + pad(tzOffset / 60) +
23 | ':' + pad(tzOffset % 60)
24 | }
25 |
26 | const coordinate = (_coordinate) => {
27 | return parseFloat(_coordinate.slice(0,2) + '.' + _coordinate.slice(2))
28 | }
29 |
30 | // follows FPTF Line, https://github.com/public-transport/friendly-public-transport-format/blob/master/spec/readme.md#line
31 | const line = (_train) => {
32 | return {
33 | type: 'line', // required
34 | id: _train['$']['tid'], // unique, required
35 | name: _train['$']['tn'], // official non-abbreviated name, required
36 | mode: modeCodes[_train['$']['type']], // see section on modes, required
37 |
38 | // seems like stops are not part of the data
39 | // routes: [], // array of route ids or route objects
40 |
41 | // operator: '123456', // operator id or operator object
42 | }
43 | }
44 |
45 | // follows FPTF Station, https://github.com/public-transport/friendly-public-transport-format/blob/master/spec/readme.md#station
46 | const station = (_station) => {
47 |
48 | // the longitudes and latitudes are optional in _station
49 | // if they are not present, the longitude and longitude should not be included in the locaton object
50 | let location = {}
51 | if (('x' in _station) && ('y' in _station)) {
52 | location = {
53 | type: 'location',
54 | longitude: coordinate(_station['x'][0]),
55 | latitude: coordinate(_station['y'][0])
56 | }
57 | } else {
58 | location = {
59 | type: 'location',
60 | }
61 | }
62 |
63 |
64 | return {
65 | type: 'station',
66 | id: _station['nr'][0],
67 | name: _station['n'][0],
68 | location: location
69 | }
70 | }
71 |
72 | // follows 'legs' in FPTF Journey, https://github.com/public-transport/friendly-public-transport-format/blob/master/spec/readme.md#journey
73 | const parseLegs = (_trainList) => {
74 |
75 | let legs = []
76 |
77 | for (let i in _trainList) {
78 | legs.push({
79 |
80 | origin: station(_trainList[i]['dep'][0]),
81 |
82 | destination: station(_trainList[i]['arr'][0]),
83 |
84 | line: line(_trainList[i]),
85 |
86 | departure: date(_trainList[i]['dep'][0]['$']),
87 |
88 | ...('ptf' in _trainList[i]['dep'][0]) && { departurePlatform: _trainList[i]['dep'][0]['ptf'][0] },
89 |
90 | arrival: date(_trainList[i]['arr'][0]['$']),
91 |
92 | ...('ptf' in _trainList[i]['arr'][0]) && { arrivalPlatform: _trainList[i]['arr'][0]['ptf'][0] },
93 |
94 | mode: modeCodes[_trainList[i]['$']['type']],
95 |
96 | public: true,
97 |
98 | // operator: '',
99 | })
100 | }
101 |
102 | return legs
103 | }
104 |
105 | // takes the base64 encoded httext field and extracts the ticket price
106 | const price = (_httext) => {
107 | let c = atob(_httext.replace(/\r?\n|\r/g, ""))
108 |
109 | let price=null;
110 | price = c.match(/Gesamtpreis:\s*\d{0,2},\d{0,2}\s*EUR/g)
111 | //when Gesamtpreis is not found search again but with html tags, because thats what the new ticket layout gives in the new systems
112 | if (!price) {
113 | const regexPrice = /Gesamtpreis: <\/span>([\d,]+,\d{2})<\/span>/;
114 | const matchPrice = c.match(regexPrice);
115 | if (matchPrice && matchPrice[1]) {
116 | price = matchPrice[1];
117 | }
118 | }else{
119 | price=price[0];
120 | }
121 | price = price.match(/\d{0,2},\d{0,2}/g)
122 | price = price[0].replace(',', '.')
123 |
124 | return parseFloat(price).toPrecision(4)
125 | }
126 |
127 | export { parseLegs, price }
128 |
--------------------------------------------------------------------------------
/lib/queryTicket.js:
--------------------------------------------------------------------------------
1 | import { parseLegs, price } from './parse.js'
2 | import requestBackend from './requestBackend.js';
3 |
4 | /**
5 | * parseTicket
6 | *
7 | * @param string | ticket API response body, JSON-decoded
8 | * @return object | the ticket data
9 | */
10 | const parseTicket = (json) => {
11 | const data = json.rporderdetails
12 |
13 | // selectors
14 | const _order = data['order'][0]['$']
15 | const _tck = data['order'][0]['tcklist']?.[0]['tck'][0]
16 | const _ticket = _tck?.['mtk'][0]
17 | let _encoded_price = _tck['htdata'][0]['ht'][0]['_']
18 | //if _tck['htdata'][0]['ht'][0]['_'] starts with data:image/png;base64, then it is a old ticket, we need to check for that
19 | if(_encoded_price.startsWith('data:image/png;base64,')){
20 | _encoded_price = _tck['htdata'][0]['ht'][1]['_'] // revert back to the old price format
21 | }
22 |
23 | const _price = _tck ? {
24 | amount: price(_encoded_price),
25 | currency: 'EUR'
26 | }: undefined
27 |
28 | return {
29 | // doesn't follow a standard by now and just dumps some info
30 | order: {
31 | ticketNumber: _order['on'],
32 | bookingDate: _order['cdt'],
33 | validFrom: _order['vfrom'],
34 | validUntil: _order['vto'],
35 | journeyStart: _order['sdt'],
36 | firstName: _ticket?.['reisender_vorname'][0],
37 | lastName: _ticket?.['reisender_nachname'][0],
38 | text: _ticket?.['txt'][0],
39 | class: _ticket?.['nvplist'][0]['nvp'][3]['_'] // could be wrong
40 | },
41 |
42 | // follows FPTF Journey, https://github.com/public-transport/friendly-public-transport-format/blob/master/spec/readme.md#journey
43 | journey: {
44 | type: 'journey',
45 | id: _order['cid'], // unique, required, use cid as id here?
46 | legs: parseLegs(data['order'][0]['schedulelist'][0]['out'][0]['trainlist'][0]['train']),
47 | price: _price
48 | },
49 |
50 | ...(data['order'][0]['schedulelist'][0]['ret'] && { returnJourney: {
51 | type: 'journey',
52 | id: _order['cid'], // unique, required, ! same as outward journey?
53 | legs: parseLegs(data['order'][0]['schedulelist'][0]['ret'][0]['trainlist'][0]['train']),
54 | price: _price
55 | }})
56 | }
57 | }
58 |
59 | /**
60 | * queryTicket function
61 | *
62 | * @param string | DB ticket number (Auftragsnummer)
63 | * @param lastName | last name of purchaser
64 | * @param kwid | kwid of the ticket (only needed for tickets from the new system)
65 | * @return Promise | fulfills with an object that contains the ticket data
66 | */
67 | const queryTicket = async (ticketNumber, lastName,kwid=null) => {
68 |
69 | let reqBody = ` `
70 |
71 | return await requestBackend(reqBody)
72 | .then(json => {
73 | return parseTicket(json)
74 | })
75 | }
76 |
77 | export {
78 | parseTicket,
79 | queryTicket,
80 | }
81 |
--------------------------------------------------------------------------------
/lib/requestBackend.js:
--------------------------------------------------------------------------------
1 | import fetch from 'node-fetch'
2 | import { parseString } from 'xml2js'
3 |
4 | const endpoint = 'https://fahrkarten.bahn.de/mobile/dbc/xs.go?'
5 |
6 | /**
7 | * requestBackend function, that is used to make a request to the backend
8 | * @param string | body of the request
9 | * @return Promise | fulfills with an object that contains the data from the request
10 | */
11 | export const requestBackend = async (reqBody) => {
12 | return fetch(endpoint, {
13 | method: 'POST',
14 | headers: {
15 | 'User-Agent': 'github.com/public-transport/db-tickets',
16 | 'Accept-Language': 'de-DE,de;q=0.9'
17 | },
18 | body: reqBody
19 | }).then(response => {
20 | if (!response.ok) {
21 | const err = new HTTPResponseError(response)
22 | err.url = endpoint
23 | err.requestBody = reqBody
24 | throw err
25 | }
26 | return response.text()
27 | }).then(body => {
28 | return new Promise((resolve, reject) => parseString(body, (err, result) => {
29 | if (err) {
30 | reject(err)
31 | }
32 | else {
33 | resolve(result)
34 | }
35 | }))
36 | }).then(json => {
37 | if (json['rperror']) {
38 | throw new XMLRPCError(json['rperror']['error'])
39 | }
40 | return json
41 | })
42 | }
43 | class HTTPResponseError extends Error {
44 | constructor(response, ...args) {
45 | super(`HTTP Error Response: ${response.status} ${response.statusText}`, ...args)
46 | this.response = response
47 | }
48 | }
49 |
50 | class XMLRPCError extends Error {
51 | constructor(response, ...args) {
52 | super(`XML Error Response:
53 | ${response[0]['$']['nr']}
54 | ${response[0]['txt'][0]}`, ...args)
55 | this.response = response
56 | }
57 | }
58 | export default requestBackend
--------------------------------------------------------------------------------
/package-lock.json:
--------------------------------------------------------------------------------
1 | {
2 | "name": "db-tickets",
3 | "version": "1.1.0",
4 | "lockfileVersion": 3,
5 | "requires": true,
6 | "packages": {
7 | "": {
8 | "name": "db-tickets",
9 | "version": "1.1.0",
10 | "license": "ISC",
11 | "dependencies": {
12 | "node-fetch": "^3.2.8",
13 | "xml2js": "^0.6.2"
14 | },
15 | "devDependencies": {
16 | "eslint": "^8.20.0"
17 | },
18 | "engines": {
19 | "node": ">=16"
20 | }
21 | },
22 | "node_modules/@aashutoshrathi/word-wrap": {
23 | "version": "1.2.6",
24 | "resolved": "https://registry.npmjs.org/@aashutoshrathi/word-wrap/-/word-wrap-1.2.6.tgz",
25 | "integrity": "sha512-1Yjs2SvM8TflER/OD3cOjhWWOZb58A2t7wpE2S9XfBYTiIl+XFhQG2bjy4Pu1I+EAlCNUzRDYDdFwFYUKvXcIA==",
26 | "dev": true,
27 | "engines": {
28 | "node": ">=0.10.0"
29 | }
30 | },
31 | "node_modules/@eslint-community/eslint-utils": {
32 | "version": "4.4.0",
33 | "resolved": "https://registry.npmjs.org/@eslint-community/eslint-utils/-/eslint-utils-4.4.0.tgz",
34 | "integrity": "sha512-1/sA4dwrzBAyeUoQ6oxahHKmrZvsnLCg4RfxW3ZFGGmQkSNQPFNLV9CUEFQP1x9EYXHTo5p6xdhZM1Ne9p/AfA==",
35 | "dev": true,
36 | "dependencies": {
37 | "eslint-visitor-keys": "^3.3.0"
38 | },
39 | "engines": {
40 | "node": "^12.22.0 || ^14.17.0 || >=16.0.0"
41 | },
42 | "peerDependencies": {
43 | "eslint": "^6.0.0 || ^7.0.0 || >=8.0.0"
44 | }
45 | },
46 | "node_modules/@eslint-community/regexpp": {
47 | "version": "4.8.0",
48 | "resolved": "https://registry.npmjs.org/@eslint-community/regexpp/-/regexpp-4.8.0.tgz",
49 | "integrity": "sha512-JylOEEzDiOryeUnFbQz+oViCXS0KsvR1mvHkoMiu5+UiBvy+RYX7tzlIIIEstF/gVa2tj9AQXk3dgnxv6KxhFg==",
50 | "dev": true,
51 | "engines": {
52 | "node": "^12.0.0 || ^14.0.0 || >=16.0.0"
53 | }
54 | },
55 | "node_modules/@eslint/eslintrc": {
56 | "version": "2.1.2",
57 | "resolved": "https://registry.npmjs.org/@eslint/eslintrc/-/eslintrc-2.1.2.tgz",
58 | "integrity": "sha512-+wvgpDsrB1YqAMdEUCcnTlpfVBH7Vqn6A/NT3D8WVXFIaKMlErPIZT3oCIAVCOtarRpMtelZLqJeU3t7WY6X6g==",
59 | "dev": true,
60 | "dependencies": {
61 | "ajv": "^6.12.4",
62 | "debug": "^4.3.2",
63 | "espree": "^9.6.0",
64 | "globals": "^13.19.0",
65 | "ignore": "^5.2.0",
66 | "import-fresh": "^3.2.1",
67 | "js-yaml": "^4.1.0",
68 | "minimatch": "^3.1.2",
69 | "strip-json-comments": "^3.1.1"
70 | },
71 | "engines": {
72 | "node": "^12.22.0 || ^14.17.0 || >=16.0.0"
73 | },
74 | "funding": {
75 | "url": "https://opencollective.com/eslint"
76 | }
77 | },
78 | "node_modules/@eslint/js": {
79 | "version": "8.49.0",
80 | "resolved": "https://registry.npmjs.org/@eslint/js/-/js-8.49.0.tgz",
81 | "integrity": "sha512-1S8uAY/MTJqVx0SC4epBq+N2yhuwtNwLbJYNZyhL2pO1ZVKn5HFXav5T41Ryzy9K9V7ZId2JB2oy/W4aCd9/2w==",
82 | "dev": true,
83 | "engines": {
84 | "node": "^12.22.0 || ^14.17.0 || >=16.0.0"
85 | }
86 | },
87 | "node_modules/@humanwhocodes/config-array": {
88 | "version": "0.11.11",
89 | "resolved": "https://registry.npmjs.org/@humanwhocodes/config-array/-/config-array-0.11.11.tgz",
90 | "integrity": "sha512-N2brEuAadi0CcdeMXUkhbZB84eskAc8MEX1By6qEchoVywSgXPIjou4rYsl0V3Hj0ZnuGycGCjdNgockbzeWNA==",
91 | "dev": true,
92 | "dependencies": {
93 | "@humanwhocodes/object-schema": "^1.2.1",
94 | "debug": "^4.1.1",
95 | "minimatch": "^3.0.5"
96 | },
97 | "engines": {
98 | "node": ">=10.10.0"
99 | }
100 | },
101 | "node_modules/@humanwhocodes/module-importer": {
102 | "version": "1.0.1",
103 | "resolved": "https://registry.npmjs.org/@humanwhocodes/module-importer/-/module-importer-1.0.1.tgz",
104 | "integrity": "sha512-bxveV4V8v5Yb4ncFTT3rPSgZBOpCkjfK0y4oVVVJwIuDVBRMDXrPyXRL988i5ap9m9bnyEEjWfm5WkBmtffLfA==",
105 | "dev": true,
106 | "engines": {
107 | "node": ">=12.22"
108 | },
109 | "funding": {
110 | "type": "github",
111 | "url": "https://github.com/sponsors/nzakas"
112 | }
113 | },
114 | "node_modules/@humanwhocodes/object-schema": {
115 | "version": "1.2.1",
116 | "resolved": "https://registry.npmjs.org/@humanwhocodes/object-schema/-/object-schema-1.2.1.tgz",
117 | "integrity": "sha512-ZnQMnLV4e7hDlUvw8H+U8ASL02SS2Gn6+9Ac3wGGLIe7+je2AeAOxPY+izIPJDfFDb7eDjev0Us8MO1iFRN8hA==",
118 | "dev": true
119 | },
120 | "node_modules/@nodelib/fs.scandir": {
121 | "version": "2.1.5",
122 | "resolved": "https://registry.npmjs.org/@nodelib/fs.scandir/-/fs.scandir-2.1.5.tgz",
123 | "integrity": "sha512-vq24Bq3ym5HEQm2NKCr3yXDwjc7vTsEThRDnkp2DK9p1uqLR+DHurm/NOTo0KG7HYHU7eppKZj3MyqYuMBf62g==",
124 | "dev": true,
125 | "dependencies": {
126 | "@nodelib/fs.stat": "2.0.5",
127 | "run-parallel": "^1.1.9"
128 | },
129 | "engines": {
130 | "node": ">= 8"
131 | }
132 | },
133 | "node_modules/@nodelib/fs.stat": {
134 | "version": "2.0.5",
135 | "resolved": "https://registry.npmjs.org/@nodelib/fs.stat/-/fs.stat-2.0.5.tgz",
136 | "integrity": "sha512-RkhPPp2zrqDAQA/2jNhnztcPAlv64XdhIp7a7454A5ovI7Bukxgt7MX7udwAu3zg1DcpPU0rz3VV1SeaqvY4+A==",
137 | "dev": true,
138 | "engines": {
139 | "node": ">= 8"
140 | }
141 | },
142 | "node_modules/@nodelib/fs.walk": {
143 | "version": "1.2.8",
144 | "resolved": "https://registry.npmjs.org/@nodelib/fs.walk/-/fs.walk-1.2.8.tgz",
145 | "integrity": "sha512-oGB+UxlgWcgQkgwo8GcEGwemoTFt3FIO9ababBmaGwXIoBKZ+GTy0pP185beGg7Llih/NSHSV2XAs1lnznocSg==",
146 | "dev": true,
147 | "dependencies": {
148 | "@nodelib/fs.scandir": "2.1.5",
149 | "fastq": "^1.6.0"
150 | },
151 | "engines": {
152 | "node": ">= 8"
153 | }
154 | },
155 | "node_modules/acorn": {
156 | "version": "8.10.0",
157 | "resolved": "https://registry.npmjs.org/acorn/-/acorn-8.10.0.tgz",
158 | "integrity": "sha512-F0SAmZ8iUtS//m8DmCTA0jlh6TDKkHQyK6xc6V4KDTyZKA9dnvX9/3sRTVQrWm79glUAZbnmmNcdYwUIHWVybw==",
159 | "dev": true,
160 | "bin": {
161 | "acorn": "bin/acorn"
162 | },
163 | "engines": {
164 | "node": ">=0.4.0"
165 | }
166 | },
167 | "node_modules/acorn-jsx": {
168 | "version": "5.3.2",
169 | "resolved": "https://registry.npmjs.org/acorn-jsx/-/acorn-jsx-5.3.2.tgz",
170 | "integrity": "sha512-rq9s+JNhf0IChjtDXxllJ7g41oZk5SlXtp0LHwyA5cejwn7vKmKp4pPri6YEePv2PU65sAsegbXtIinmDFDXgQ==",
171 | "dev": true,
172 | "peerDependencies": {
173 | "acorn": "^6.0.0 || ^7.0.0 || ^8.0.0"
174 | }
175 | },
176 | "node_modules/ajv": {
177 | "version": "6.12.6",
178 | "resolved": "https://registry.npmjs.org/ajv/-/ajv-6.12.6.tgz",
179 | "integrity": "sha512-j3fVLgvTo527anyYyJOGTYJbG+vnnQYvE0m5mmkc1TK+nxAppkCLMIL0aZ4dblVCNoGShhm+kzE4ZUykBoMg4g==",
180 | "dev": true,
181 | "dependencies": {
182 | "fast-deep-equal": "^3.1.1",
183 | "fast-json-stable-stringify": "^2.0.0",
184 | "json-schema-traverse": "^0.4.1",
185 | "uri-js": "^4.2.2"
186 | },
187 | "funding": {
188 | "type": "github",
189 | "url": "https://github.com/sponsors/epoberezkin"
190 | }
191 | },
192 | "node_modules/ansi-regex": {
193 | "version": "5.0.1",
194 | "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.1.tgz",
195 | "integrity": "sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ==",
196 | "dev": true,
197 | "engines": {
198 | "node": ">=8"
199 | }
200 | },
201 | "node_modules/ansi-styles": {
202 | "version": "4.3.0",
203 | "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz",
204 | "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==",
205 | "dev": true,
206 | "dependencies": {
207 | "color-convert": "^2.0.1"
208 | },
209 | "engines": {
210 | "node": ">=8"
211 | },
212 | "funding": {
213 | "url": "https://github.com/chalk/ansi-styles?sponsor=1"
214 | }
215 | },
216 | "node_modules/argparse": {
217 | "version": "2.0.1",
218 | "resolved": "https://registry.npmjs.org/argparse/-/argparse-2.0.1.tgz",
219 | "integrity": "sha512-8+9WqebbFzpX9OR+Wa6O29asIogeRMzcGtAINdpMHHyAg10f05aSFVBbcEqGf/PXw1EjAZ+q2/bEBg3DvurK3Q==",
220 | "dev": true
221 | },
222 | "node_modules/balanced-match": {
223 | "version": "1.0.2",
224 | "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.2.tgz",
225 | "integrity": "sha512-3oSeUO0TMV67hN1AmbXsK4yaqU7tjiHlbxRDZOpH0KW9+CeX4bRAaX0Anxt0tx2MrpRpWwQaPwIlISEJhYU5Pw==",
226 | "dev": true
227 | },
228 | "node_modules/brace-expansion": {
229 | "version": "1.1.11",
230 | "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.11.tgz",
231 | "integrity": "sha512-iCuPHDFgrHX7H2vEI/5xpz07zSHB00TpugqhmYtVmMO6518mCuRMoOYFldEBl0g187ufozdaHgWKcYFb61qGiA==",
232 | "dev": true,
233 | "dependencies": {
234 | "balanced-match": "^1.0.0",
235 | "concat-map": "0.0.1"
236 | }
237 | },
238 | "node_modules/callsites": {
239 | "version": "3.1.0",
240 | "resolved": "https://registry.npmjs.org/callsites/-/callsites-3.1.0.tgz",
241 | "integrity": "sha512-P8BjAsXvZS+VIDUI11hHCQEv74YT67YUi5JJFNWIqL235sBmjX4+qx9Muvls5ivyNENctx46xQLQ3aTuE7ssaQ==",
242 | "dev": true,
243 | "engines": {
244 | "node": ">=6"
245 | }
246 | },
247 | "node_modules/chalk": {
248 | "version": "4.1.2",
249 | "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz",
250 | "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==",
251 | "dev": true,
252 | "dependencies": {
253 | "ansi-styles": "^4.1.0",
254 | "supports-color": "^7.1.0"
255 | },
256 | "engines": {
257 | "node": ">=10"
258 | },
259 | "funding": {
260 | "url": "https://github.com/chalk/chalk?sponsor=1"
261 | }
262 | },
263 | "node_modules/color-convert": {
264 | "version": "2.0.1",
265 | "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz",
266 | "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==",
267 | "dev": true,
268 | "dependencies": {
269 | "color-name": "~1.1.4"
270 | },
271 | "engines": {
272 | "node": ">=7.0.0"
273 | }
274 | },
275 | "node_modules/color-name": {
276 | "version": "1.1.4",
277 | "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz",
278 | "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==",
279 | "dev": true
280 | },
281 | "node_modules/concat-map": {
282 | "version": "0.0.1",
283 | "resolved": "https://registry.npmjs.org/concat-map/-/concat-map-0.0.1.tgz",
284 | "integrity": "sha512-/Srv4dswyQNBfohGpz9o6Yb3Gz3SrUDqBH5rTuhGR7ahtlbYKnVxw2bCFMRljaA7EXHaXZ8wsHdodFvbkhKmqg==",
285 | "dev": true
286 | },
287 | "node_modules/cross-spawn": {
288 | "version": "7.0.3",
289 | "resolved": "https://registry.npmjs.org/cross-spawn/-/cross-spawn-7.0.3.tgz",
290 | "integrity": "sha512-iRDPJKUPVEND7dHPO8rkbOnPpyDygcDFtWjpeWNCgy8WP2rXcxXL8TskReQl6OrB2G7+UJrags1q15Fudc7G6w==",
291 | "dev": true,
292 | "dependencies": {
293 | "path-key": "^3.1.0",
294 | "shebang-command": "^2.0.0",
295 | "which": "^2.0.1"
296 | },
297 | "engines": {
298 | "node": ">= 8"
299 | }
300 | },
301 | "node_modules/data-uri-to-buffer": {
302 | "version": "4.0.1",
303 | "resolved": "https://registry.npmjs.org/data-uri-to-buffer/-/data-uri-to-buffer-4.0.1.tgz",
304 | "integrity": "sha512-0R9ikRb668HB7QDxT1vkpuUBtqc53YyAwMwGeUFKRojY/NWKvdZ+9UYtRfGmhqNbRkTSVpMbmyhXipFFv2cb/A==",
305 | "engines": {
306 | "node": ">= 12"
307 | }
308 | },
309 | "node_modules/debug": {
310 | "version": "4.3.4",
311 | "resolved": "https://registry.npmjs.org/debug/-/debug-4.3.4.tgz",
312 | "integrity": "sha512-PRWFHuSU3eDtQJPvnNY7Jcket1j0t5OuOsFzPPzsekD52Zl8qUfFIPEiswXqIvHWGVHOgX+7G/vCNNhehwxfkQ==",
313 | "dev": true,
314 | "dependencies": {
315 | "ms": "2.1.2"
316 | },
317 | "engines": {
318 | "node": ">=6.0"
319 | },
320 | "peerDependenciesMeta": {
321 | "supports-color": {
322 | "optional": true
323 | }
324 | }
325 | },
326 | "node_modules/deep-is": {
327 | "version": "0.1.4",
328 | "resolved": "https://registry.npmjs.org/deep-is/-/deep-is-0.1.4.tgz",
329 | "integrity": "sha512-oIPzksmTg4/MriiaYGO+okXDT7ztn/w3Eptv/+gSIdMdKsJo0u4CfYNFJPy+4SKMuCqGw2wxnA+URMg3t8a/bQ==",
330 | "dev": true
331 | },
332 | "node_modules/doctrine": {
333 | "version": "3.0.0",
334 | "resolved": "https://registry.npmjs.org/doctrine/-/doctrine-3.0.0.tgz",
335 | "integrity": "sha512-yS+Q5i3hBf7GBkd4KG8a7eBNNWNGLTaEwwYWUijIYM7zrlYDM0BFXHjjPWlWZ1Rg7UaddZeIDmi9jF3HmqiQ2w==",
336 | "dev": true,
337 | "dependencies": {
338 | "esutils": "^2.0.2"
339 | },
340 | "engines": {
341 | "node": ">=6.0.0"
342 | }
343 | },
344 | "node_modules/escape-string-regexp": {
345 | "version": "4.0.0",
346 | "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-4.0.0.tgz",
347 | "integrity": "sha512-TtpcNJ3XAzx3Gq8sWRzJaVajRs0uVxA2YAkdb1jm2YkPz4G6egUFAyA3n5vtEIZefPk5Wa4UXbKuS5fKkJWdgA==",
348 | "dev": true,
349 | "engines": {
350 | "node": ">=10"
351 | },
352 | "funding": {
353 | "url": "https://github.com/sponsors/sindresorhus"
354 | }
355 | },
356 | "node_modules/eslint": {
357 | "version": "8.49.0",
358 | "resolved": "https://registry.npmjs.org/eslint/-/eslint-8.49.0.tgz",
359 | "integrity": "sha512-jw03ENfm6VJI0jA9U+8H5zfl5b+FvuU3YYvZRdZHOlU2ggJkxrlkJH4HcDrZpj6YwD8kuYqvQM8LyesoazrSOQ==",
360 | "dev": true,
361 | "dependencies": {
362 | "@eslint-community/eslint-utils": "^4.2.0",
363 | "@eslint-community/regexpp": "^4.6.1",
364 | "@eslint/eslintrc": "^2.1.2",
365 | "@eslint/js": "8.49.0",
366 | "@humanwhocodes/config-array": "^0.11.11",
367 | "@humanwhocodes/module-importer": "^1.0.1",
368 | "@nodelib/fs.walk": "^1.2.8",
369 | "ajv": "^6.12.4",
370 | "chalk": "^4.0.0",
371 | "cross-spawn": "^7.0.2",
372 | "debug": "^4.3.2",
373 | "doctrine": "^3.0.0",
374 | "escape-string-regexp": "^4.0.0",
375 | "eslint-scope": "^7.2.2",
376 | "eslint-visitor-keys": "^3.4.3",
377 | "espree": "^9.6.1",
378 | "esquery": "^1.4.2",
379 | "esutils": "^2.0.2",
380 | "fast-deep-equal": "^3.1.3",
381 | "file-entry-cache": "^6.0.1",
382 | "find-up": "^5.0.0",
383 | "glob-parent": "^6.0.2",
384 | "globals": "^13.19.0",
385 | "graphemer": "^1.4.0",
386 | "ignore": "^5.2.0",
387 | "imurmurhash": "^0.1.4",
388 | "is-glob": "^4.0.0",
389 | "is-path-inside": "^3.0.3",
390 | "js-yaml": "^4.1.0",
391 | "json-stable-stringify-without-jsonify": "^1.0.1",
392 | "levn": "^0.4.1",
393 | "lodash.merge": "^4.6.2",
394 | "minimatch": "^3.1.2",
395 | "natural-compare": "^1.4.0",
396 | "optionator": "^0.9.3",
397 | "strip-ansi": "^6.0.1",
398 | "text-table": "^0.2.0"
399 | },
400 | "bin": {
401 | "eslint": "bin/eslint.js"
402 | },
403 | "engines": {
404 | "node": "^12.22.0 || ^14.17.0 || >=16.0.0"
405 | },
406 | "funding": {
407 | "url": "https://opencollective.com/eslint"
408 | }
409 | },
410 | "node_modules/eslint-scope": {
411 | "version": "7.2.2",
412 | "resolved": "https://registry.npmjs.org/eslint-scope/-/eslint-scope-7.2.2.tgz",
413 | "integrity": "sha512-dOt21O7lTMhDM+X9mB4GX+DZrZtCUJPL/wlcTqxyrx5IvO0IYtILdtrQGQp+8n5S0gwSVmOf9NQrjMOgfQZlIg==",
414 | "dev": true,
415 | "dependencies": {
416 | "esrecurse": "^4.3.0",
417 | "estraverse": "^5.2.0"
418 | },
419 | "engines": {
420 | "node": "^12.22.0 || ^14.17.0 || >=16.0.0"
421 | },
422 | "funding": {
423 | "url": "https://opencollective.com/eslint"
424 | }
425 | },
426 | "node_modules/eslint-visitor-keys": {
427 | "version": "3.4.3",
428 | "resolved": "https://registry.npmjs.org/eslint-visitor-keys/-/eslint-visitor-keys-3.4.3.tgz",
429 | "integrity": "sha512-wpc+LXeiyiisxPlEkUzU6svyS1frIO3Mgxj1fdy7Pm8Ygzguax2N3Fa/D/ag1WqbOprdI+uY6wMUl8/a2G+iag==",
430 | "dev": true,
431 | "engines": {
432 | "node": "^12.22.0 || ^14.17.0 || >=16.0.0"
433 | },
434 | "funding": {
435 | "url": "https://opencollective.com/eslint"
436 | }
437 | },
438 | "node_modules/espree": {
439 | "version": "9.6.1",
440 | "resolved": "https://registry.npmjs.org/espree/-/espree-9.6.1.tgz",
441 | "integrity": "sha512-oruZaFkjorTpF32kDSI5/75ViwGeZginGGy2NoOSg3Q9bnwlnmDm4HLnkl0RE3n+njDXR037aY1+x58Z/zFdwQ==",
442 | "dev": true,
443 | "dependencies": {
444 | "acorn": "^8.9.0",
445 | "acorn-jsx": "^5.3.2",
446 | "eslint-visitor-keys": "^3.4.1"
447 | },
448 | "engines": {
449 | "node": "^12.22.0 || ^14.17.0 || >=16.0.0"
450 | },
451 | "funding": {
452 | "url": "https://opencollective.com/eslint"
453 | }
454 | },
455 | "node_modules/esquery": {
456 | "version": "1.5.0",
457 | "resolved": "https://registry.npmjs.org/esquery/-/esquery-1.5.0.tgz",
458 | "integrity": "sha512-YQLXUplAwJgCydQ78IMJywZCceoqk1oH01OERdSAJc/7U2AylwjhSCLDEtqwg811idIS/9fIU5GjG73IgjKMVg==",
459 | "dev": true,
460 | "dependencies": {
461 | "estraverse": "^5.1.0"
462 | },
463 | "engines": {
464 | "node": ">=0.10"
465 | }
466 | },
467 | "node_modules/esrecurse": {
468 | "version": "4.3.0",
469 | "resolved": "https://registry.npmjs.org/esrecurse/-/esrecurse-4.3.0.tgz",
470 | "integrity": "sha512-KmfKL3b6G+RXvP8N1vr3Tq1kL/oCFgn2NYXEtqP8/L3pKapUA4G8cFVaoF3SU323CD4XypR/ffioHmkti6/Tag==",
471 | "dev": true,
472 | "dependencies": {
473 | "estraverse": "^5.2.0"
474 | },
475 | "engines": {
476 | "node": ">=4.0"
477 | }
478 | },
479 | "node_modules/estraverse": {
480 | "version": "5.3.0",
481 | "resolved": "https://registry.npmjs.org/estraverse/-/estraverse-5.3.0.tgz",
482 | "integrity": "sha512-MMdARuVEQziNTeJD8DgMqmhwR11BRQ/cBP+pLtYdSTnf3MIO8fFeiINEbX36ZdNlfU/7A9f3gUw49B3oQsvwBA==",
483 | "dev": true,
484 | "engines": {
485 | "node": ">=4.0"
486 | }
487 | },
488 | "node_modules/esutils": {
489 | "version": "2.0.3",
490 | "resolved": "https://registry.npmjs.org/esutils/-/esutils-2.0.3.tgz",
491 | "integrity": "sha512-kVscqXk4OCp68SZ0dkgEKVi6/8ij300KBWTJq32P/dYeWTSwK41WyTxalN1eRmA5Z9UU/LX9D7FWSmV9SAYx6g==",
492 | "dev": true,
493 | "engines": {
494 | "node": ">=0.10.0"
495 | }
496 | },
497 | "node_modules/fast-deep-equal": {
498 | "version": "3.1.3",
499 | "resolved": "https://registry.npmjs.org/fast-deep-equal/-/fast-deep-equal-3.1.3.tgz",
500 | "integrity": "sha512-f3qQ9oQy9j2AhBe/H9VC91wLmKBCCU/gDOnKNAYG5hswO7BLKj09Hc5HYNz9cGI++xlpDCIgDaitVs03ATR84Q==",
501 | "dev": true
502 | },
503 | "node_modules/fast-json-stable-stringify": {
504 | "version": "2.1.0",
505 | "resolved": "https://registry.npmjs.org/fast-json-stable-stringify/-/fast-json-stable-stringify-2.1.0.tgz",
506 | "integrity": "sha512-lhd/wF+Lk98HZoTCtlVraHtfh5XYijIjalXck7saUtuanSDyLMxnHhSXEDJqHxD7msR8D0uCmqlkwjCV8xvwHw==",
507 | "dev": true
508 | },
509 | "node_modules/fast-levenshtein": {
510 | "version": "2.0.6",
511 | "resolved": "https://registry.npmjs.org/fast-levenshtein/-/fast-levenshtein-2.0.6.tgz",
512 | "integrity": "sha512-DCXu6Ifhqcks7TZKY3Hxp3y6qphY5SJZmrWMDrKcERSOXWQdMhU9Ig/PYrzyw/ul9jOIyh0N4M0tbC5hodg8dw==",
513 | "dev": true
514 | },
515 | "node_modules/fastq": {
516 | "version": "1.15.0",
517 | "resolved": "https://registry.npmjs.org/fastq/-/fastq-1.15.0.tgz",
518 | "integrity": "sha512-wBrocU2LCXXa+lWBt8RoIRD89Fi8OdABODa/kEnyeyjS5aZO5/GNvI5sEINADqP/h8M29UHTHUb53sUu5Ihqdw==",
519 | "dev": true,
520 | "dependencies": {
521 | "reusify": "^1.0.4"
522 | }
523 | },
524 | "node_modules/fetch-blob": {
525 | "version": "3.2.0",
526 | "resolved": "https://registry.npmjs.org/fetch-blob/-/fetch-blob-3.2.0.tgz",
527 | "integrity": "sha512-7yAQpD2UMJzLi1Dqv7qFYnPbaPx7ZfFK6PiIxQ4PfkGPyNyl2Ugx+a/umUonmKqjhM4DnfbMvdX6otXq83soQQ==",
528 | "funding": [
529 | {
530 | "type": "github",
531 | "url": "https://github.com/sponsors/jimmywarting"
532 | },
533 | {
534 | "type": "paypal",
535 | "url": "https://paypal.me/jimmywarting"
536 | }
537 | ],
538 | "dependencies": {
539 | "node-domexception": "^1.0.0",
540 | "web-streams-polyfill": "^3.0.3"
541 | },
542 | "engines": {
543 | "node": "^12.20 || >= 14.13"
544 | }
545 | },
546 | "node_modules/file-entry-cache": {
547 | "version": "6.0.1",
548 | "resolved": "https://registry.npmjs.org/file-entry-cache/-/file-entry-cache-6.0.1.tgz",
549 | "integrity": "sha512-7Gps/XWymbLk2QLYK4NzpMOrYjMhdIxXuIvy2QBsLE6ljuodKvdkWs/cpyJJ3CVIVpH0Oi1Hvg1ovbMzLdFBBg==",
550 | "dev": true,
551 | "dependencies": {
552 | "flat-cache": "^3.0.4"
553 | },
554 | "engines": {
555 | "node": "^10.12.0 || >=12.0.0"
556 | }
557 | },
558 | "node_modules/find-up": {
559 | "version": "5.0.0",
560 | "resolved": "https://registry.npmjs.org/find-up/-/find-up-5.0.0.tgz",
561 | "integrity": "sha512-78/PXT1wlLLDgTzDs7sjq9hzz0vXD+zn+7wypEe4fXQxCmdmqfGsEPQxmiCSQI3ajFV91bVSsvNtrJRiW6nGng==",
562 | "dev": true,
563 | "dependencies": {
564 | "locate-path": "^6.0.0",
565 | "path-exists": "^4.0.0"
566 | },
567 | "engines": {
568 | "node": ">=10"
569 | },
570 | "funding": {
571 | "url": "https://github.com/sponsors/sindresorhus"
572 | }
573 | },
574 | "node_modules/flat-cache": {
575 | "version": "3.1.0",
576 | "resolved": "https://registry.npmjs.org/flat-cache/-/flat-cache-3.1.0.tgz",
577 | "integrity": "sha512-OHx4Qwrrt0E4jEIcI5/Xb+f+QmJYNj2rrK8wiIdQOIrB9WrrJL8cjZvXdXuBTkkEwEqLycb5BeZDV1o2i9bTew==",
578 | "dev": true,
579 | "dependencies": {
580 | "flatted": "^3.2.7",
581 | "keyv": "^4.5.3",
582 | "rimraf": "^3.0.2"
583 | },
584 | "engines": {
585 | "node": ">=12.0.0"
586 | }
587 | },
588 | "node_modules/flatted": {
589 | "version": "3.2.7",
590 | "resolved": "https://registry.npmjs.org/flatted/-/flatted-3.2.7.tgz",
591 | "integrity": "sha512-5nqDSxl8nn5BSNxyR3n4I6eDmbolI6WT+QqR547RwxQapgjQBmtktdP+HTBb/a/zLsbzERTONyUB5pefh5TtjQ==",
592 | "dev": true
593 | },
594 | "node_modules/formdata-polyfill": {
595 | "version": "4.0.10",
596 | "resolved": "https://registry.npmjs.org/formdata-polyfill/-/formdata-polyfill-4.0.10.tgz",
597 | "integrity": "sha512-buewHzMvYL29jdeQTVILecSaZKnt/RJWjoZCF5OW60Z67/GmSLBkOFM7qh1PI3zFNtJbaZL5eQu1vLfazOwj4g==",
598 | "dependencies": {
599 | "fetch-blob": "^3.1.2"
600 | },
601 | "engines": {
602 | "node": ">=12.20.0"
603 | }
604 | },
605 | "node_modules/fs.realpath": {
606 | "version": "1.0.0",
607 | "resolved": "https://registry.npmjs.org/fs.realpath/-/fs.realpath-1.0.0.tgz",
608 | "integrity": "sha512-OO0pH2lK6a0hZnAdau5ItzHPI6pUlvI7jMVnxUQRtw4owF2wk8lOSabtGDCTP4Ggrg2MbGnWO9X8K1t4+fGMDw==",
609 | "dev": true
610 | },
611 | "node_modules/glob": {
612 | "version": "7.2.3",
613 | "resolved": "https://registry.npmjs.org/glob/-/glob-7.2.3.tgz",
614 | "integrity": "sha512-nFR0zLpU2YCaRxwoCJvL6UvCH2JFyFVIvwTLsIf21AuHlMskA1hhTdk+LlYJtOlYt9v6dvszD2BGRqBL+iQK9Q==",
615 | "dev": true,
616 | "dependencies": {
617 | "fs.realpath": "^1.0.0",
618 | "inflight": "^1.0.4",
619 | "inherits": "2",
620 | "minimatch": "^3.1.1",
621 | "once": "^1.3.0",
622 | "path-is-absolute": "^1.0.0"
623 | },
624 | "engines": {
625 | "node": "*"
626 | },
627 | "funding": {
628 | "url": "https://github.com/sponsors/isaacs"
629 | }
630 | },
631 | "node_modules/glob-parent": {
632 | "version": "6.0.2",
633 | "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-6.0.2.tgz",
634 | "integrity": "sha512-XxwI8EOhVQgWp6iDL+3b0r86f4d6AX6zSU55HfB4ydCEuXLXc5FcYeOu+nnGftS4TEju/11rt4KJPTMgbfmv4A==",
635 | "dev": true,
636 | "dependencies": {
637 | "is-glob": "^4.0.3"
638 | },
639 | "engines": {
640 | "node": ">=10.13.0"
641 | }
642 | },
643 | "node_modules/globals": {
644 | "version": "13.21.0",
645 | "resolved": "https://registry.npmjs.org/globals/-/globals-13.21.0.tgz",
646 | "integrity": "sha512-ybyme3s4yy/t/3s35bewwXKOf7cvzfreG2lH0lZl0JB7I4GxRP2ghxOK/Nb9EkRXdbBXZLfq/p/0W2JUONB/Gg==",
647 | "dev": true,
648 | "dependencies": {
649 | "type-fest": "^0.20.2"
650 | },
651 | "engines": {
652 | "node": ">=8"
653 | },
654 | "funding": {
655 | "url": "https://github.com/sponsors/sindresorhus"
656 | }
657 | },
658 | "node_modules/graphemer": {
659 | "version": "1.4.0",
660 | "resolved": "https://registry.npmjs.org/graphemer/-/graphemer-1.4.0.tgz",
661 | "integrity": "sha512-EtKwoO6kxCL9WO5xipiHTZlSzBm7WLT627TqC/uVRd0HKmq8NXyebnNYxDoBi7wt8eTWrUrKXCOVaFq9x1kgag==",
662 | "dev": true
663 | },
664 | "node_modules/has-flag": {
665 | "version": "4.0.0",
666 | "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz",
667 | "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==",
668 | "dev": true,
669 | "engines": {
670 | "node": ">=8"
671 | }
672 | },
673 | "node_modules/ignore": {
674 | "version": "5.2.4",
675 | "resolved": "https://registry.npmjs.org/ignore/-/ignore-5.2.4.tgz",
676 | "integrity": "sha512-MAb38BcSbH0eHNBxn7ql2NH/kX33OkB3lZ1BNdh7ENeRChHTYsTvWrMubiIAMNS2llXEEgZ1MUOBtXChP3kaFQ==",
677 | "dev": true,
678 | "engines": {
679 | "node": ">= 4"
680 | }
681 | },
682 | "node_modules/import-fresh": {
683 | "version": "3.3.0",
684 | "resolved": "https://registry.npmjs.org/import-fresh/-/import-fresh-3.3.0.tgz",
685 | "integrity": "sha512-veYYhQa+D1QBKznvhUHxb8faxlrwUnxseDAbAp457E0wLNio2bOSKnjYDhMj+YiAq61xrMGhQk9iXVk5FzgQMw==",
686 | "dev": true,
687 | "dependencies": {
688 | "parent-module": "^1.0.0",
689 | "resolve-from": "^4.0.0"
690 | },
691 | "engines": {
692 | "node": ">=6"
693 | },
694 | "funding": {
695 | "url": "https://github.com/sponsors/sindresorhus"
696 | }
697 | },
698 | "node_modules/imurmurhash": {
699 | "version": "0.1.4",
700 | "resolved": "https://registry.npmjs.org/imurmurhash/-/imurmurhash-0.1.4.tgz",
701 | "integrity": "sha512-JmXMZ6wuvDmLiHEml9ykzqO6lwFbof0GG4IkcGaENdCRDDmMVnny7s5HsIgHCbaq0w2MyPhDqkhTUgS2LU2PHA==",
702 | "dev": true,
703 | "engines": {
704 | "node": ">=0.8.19"
705 | }
706 | },
707 | "node_modules/inflight": {
708 | "version": "1.0.6",
709 | "resolved": "https://registry.npmjs.org/inflight/-/inflight-1.0.6.tgz",
710 | "integrity": "sha512-k92I/b08q4wvFscXCLvqfsHCrjrF7yiXsQuIVvVE7N82W3+aqpzuUdBbfhWcy/FZR3/4IgflMgKLOsvPDrGCJA==",
711 | "dev": true,
712 | "dependencies": {
713 | "once": "^1.3.0",
714 | "wrappy": "1"
715 | }
716 | },
717 | "node_modules/inherits": {
718 | "version": "2.0.4",
719 | "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.4.tgz",
720 | "integrity": "sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ==",
721 | "dev": true
722 | "integrity": "sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ==",
723 | "dev": true
724 | },
725 | "node_modules/is-extglob": {
726 | "version": "2.1.1",
727 | "resolved": "https://registry.npmjs.org/is-extglob/-/is-extglob-2.1.1.tgz",
728 | "integrity": "sha512-SbKbANkN603Vi4jEZv49LeVJMn4yGwsbzZworEoyEiutsN3nJYdbO36zfhGJ6QEDpOZIFkDtnq5JRxmvl3jsoQ==",
729 | "dev": true,
730 | "engines": {
731 | "node": ">=0.10.0"
732 | }
733 | },
734 | "node_modules/is-glob": {
735 | "version": "4.0.3",
736 | "resolved": "https://registry.npmjs.org/is-glob/-/is-glob-4.0.3.tgz",
737 | "integrity": "sha512-xelSayHH36ZgE7ZWhli7pW34hNbNl8Ojv5KVmkJD4hBdD3th8Tfk9vYasLM+mXWOZhFkgZfxhLSnrwRr4elSSg==",
738 | "dev": true,
739 | "dependencies": {
740 | "is-extglob": "^2.1.1"
741 | },
742 | "engines": {
743 | "node": ">=0.10.0"
744 | }
745 | },
746 | "node_modules/is-path-inside": {
747 | "version": "3.0.3",
748 | "resolved": "https://registry.npmjs.org/is-path-inside/-/is-path-inside-3.0.3.tgz",
749 | "integrity": "sha512-Fd4gABb+ycGAmKou8eMftCupSir5lRxqf4aD/vd0cD2qc4HL07OjCeuHMr8Ro4CoMaeCKDB0/ECBOVWjTwUvPQ==",
750 | "dev": true,
751 | "engines": {
752 | "node": ">=8"
753 | }
754 | },
755 | "node_modules/isexe": {
756 | "version": "2.0.0",
757 | "resolved": "https://registry.npmjs.org/isexe/-/isexe-2.0.0.tgz",
758 | "integrity": "sha512-RHxMLp9lnKHGHRng9QFhRCMbYAcVpn69smSGcq3f36xjgVVWThj4qqLbTLlq7Ssj8B+fIQ1EuCEGI2lKsyQeIw==",
759 | "dev": true
760 | },
761 | "node_modules/js-yaml": {
762 | "version": "4.1.0",
763 | "resolved": "https://registry.npmjs.org/js-yaml/-/js-yaml-4.1.0.tgz",
764 | "integrity": "sha512-wpxZs9NoxZaJESJGIZTyDEaYpl0FKSA+FB9aJiyemKhMwkxQg63h4T1KJgUGHpTqPDNRcmmYLugrRjJlBtWvRA==",
765 | "dev": true,
766 | "dependencies": {
767 | "argparse": "^2.0.1"
768 | },
769 | "bin": {
770 | "js-yaml": "bin/js-yaml.js"
771 | }
772 | },
773 | "node_modules/json-buffer": {
774 | "version": "3.0.1",
775 | "resolved": "https://registry.npmjs.org/json-buffer/-/json-buffer-3.0.1.tgz",
776 | "integrity": "sha512-4bV5BfR2mqfQTJm+V5tPPdf+ZpuhiIvTuAB5g8kcrXOZpTT/QwwVRWBywX1ozr6lEuPdbHxwaJlm9G6mI2sfSQ==",
777 | "dev": true
778 | },
779 | "node_modules/json-schema-traverse": {
780 | "version": "0.4.1",
781 | "resolved": "https://registry.npmjs.org/json-schema-traverse/-/json-schema-traverse-0.4.1.tgz",
782 | "integrity": "sha512-xbbCH5dCYU5T8LcEhhuh7HJ88HXuW3qsI3Y0zOZFKfZEHcpWiHU/Jxzk629Brsab/mMiHQti9wMP+845RPe3Vg==",
783 | "dev": true
784 | },
785 | "node_modules/json-stable-stringify-without-jsonify": {
786 | "version": "1.0.1",
787 | "resolved": "https://registry.npmjs.org/json-stable-stringify-without-jsonify/-/json-stable-stringify-without-jsonify-1.0.1.tgz",
788 | "integrity": "sha512-Bdboy+l7tA3OGW6FjyFHWkP5LuByj1Tk33Ljyq0axyzdk9//JSi2u3fP1QSmd1KNwq6VOKYGlAu87CisVir6Pw==",
789 | "dev": true
790 | },
791 | "node_modules/keyv": {
792 | "version": "4.5.3",
793 | "resolved": "https://registry.npmjs.org/keyv/-/keyv-4.5.3.tgz",
794 | "integrity": "sha512-QCiSav9WaX1PgETJ+SpNnx2PRRapJ/oRSXM4VO5OGYGSjrxbKPVFVhB3l2OCbLCk329N8qyAtsJjSjvVBWzEug==",
795 | "dev": true,
796 | "dependencies": {
797 | "json-buffer": "3.0.1"
798 | }
799 | },
800 | "node_modules/levn": {
801 | "version": "0.4.1",
802 | "resolved": "https://registry.npmjs.org/levn/-/levn-0.4.1.tgz",
803 | "integrity": "sha512-+bT2uH4E5LGE7h/n3evcS/sQlJXCpIp6ym8OWJ5eV6+67Dsql/LaaT7qJBAt2rzfoa/5QBGBhxDix1dMt2kQKQ==",
804 | "dev": true,
805 | "dependencies": {
806 | "prelude-ls": "^1.2.1",
807 | "type-check": "~0.4.0"
808 | },
809 | "engines": {
810 | "node": ">= 0.8.0"
811 | }
812 | },
813 | "node_modules/locate-path": {
814 | "version": "6.0.0",
815 | "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-6.0.0.tgz",
816 | "integrity": "sha512-iPZK6eYjbxRu3uB4/WZ3EsEIMJFMqAoopl3R+zuq0UjcAm/MO6KCweDgPfP3elTztoKP3KtnVHxTn2NHBSDVUw==",
817 | "dev": true,
818 | "dependencies": {
819 | "p-locate": "^5.0.0"
820 | },
821 | "engines": {
822 | "node": ">=10"
823 | },
824 | "funding": {
825 | "url": "https://github.com/sponsors/sindresorhus"
826 | }
827 | },
828 | "node_modules/lodash.merge": {
829 | "version": "4.6.2",
830 | "resolved": "https://registry.npmjs.org/lodash.merge/-/lodash.merge-4.6.2.tgz",
831 | "integrity": "sha512-0KpjqXRVvrYyCsX1swR/XTK0va6VQkQM6MNo7PqW77ByjAhoARA8EfrP1N4+KlKj8YS0ZUCtRT/YUuhyYDujIQ==",
832 | "dev": true
833 | },
834 | "node_modules/minimatch": {
835 | "version": "3.1.2",
836 | "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.1.2.tgz",
837 | "integrity": "sha512-J7p63hRiAjw1NDEww1W7i37+ByIrOWO5XQQAzZ3VOcL0PNybwpfmV/N05zFAzwQ9USyEcX6t3UO+K5aqBQOIHw==",
838 | "dev": true,
839 | "dependencies": {
840 | "brace-expansion": "^1.1.7"
841 | },
842 | "engines": {
843 | "node": "*"
844 | }
845 | },
846 | "node_modules/ms": {
847 | "version": "2.1.2",
848 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.2.tgz",
849 | "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==",
850 | "dev": true
851 | },
852 | "node_modules/natural-compare": {
853 | "version": "1.4.0",
854 | "resolved": "https://registry.npmjs.org/natural-compare/-/natural-compare-1.4.0.tgz",
855 | "integrity": "sha512-OWND8ei3VtNC9h7V60qff3SVobHr996CTwgxubgyQYEpg290h9J0buyECNNJexkFm5sOajh5G116RYA1c8ZMSw==",
856 | "dev": true
857 | },
858 | "node_modules/node-domexception": {
859 | "version": "1.0.0",
860 | "resolved": "https://registry.npmjs.org/node-domexception/-/node-domexception-1.0.0.tgz",
861 | "integrity": "sha512-/jKZoMpw0F8GRwl4/eLROPA3cfcXtLApP0QzLmUT/HuPCZWyB7IY9ZrMeKw2O/nFIqPQB3PVM9aYm0F312AXDQ==",
862 | "funding": [
863 | {
864 | "type": "github",
865 | "url": "https://github.com/sponsors/jimmywarting"
866 | },
867 | {
868 | "type": "github",
869 | "url": "https://paypal.me/jimmywarting"
870 | }
871 | ],
872 | "engines": {
873 | "node": ">=10.5.0"
874 | }
875 | },
876 | "node_modules/node-fetch": {
877 | "version": "3.3.2",
878 | "resolved": "https://registry.npmjs.org/node-fetch/-/node-fetch-3.3.2.tgz",
879 | "integrity": "sha512-dRB78srN/l6gqWulah9SrxeYnxeddIG30+GOqK/9OlLVyLg3HPnr6SqOWTWOXKRwC2eGYCkZ59NNuSgvSrpgOA==",
880 | "dependencies": {
881 | "data-uri-to-buffer": "^4.0.0",
882 | "fetch-blob": "^3.1.4",
883 | "formdata-polyfill": "^4.0.10"
884 | },
885 | "engines": {
886 | "node": "^12.20.0 || ^14.13.1 || >=16.0.0"
887 | },
888 | "funding": {
889 | "type": "opencollective",
890 | "url": "https://opencollective.com/node-fetch"
891 | }
892 | },
893 | "node_modules/once": {
894 | "version": "1.4.0",
895 | "resolved": "https://registry.npmjs.org/once/-/once-1.4.0.tgz",
896 | "integrity": "sha512-lNaJgI+2Q5URQBkccEKHTQOPaXdUxnZZElQTZY0MFUAuaEqe1E+Nyvgdz/aIyNi6Z9MzO5dv1H8n58/GELp3+w==",
897 | "dev": true,
898 | "dependencies": {
899 | "wrappy": "1"
900 | }
901 | },
902 | "node_modules/optionator": {
903 | "version": "0.9.3",
904 | "resolved": "https://registry.npmjs.org/optionator/-/optionator-0.9.3.tgz",
905 | "integrity": "sha512-JjCoypp+jKn1ttEFExxhetCKeJt9zhAgAve5FXHixTvFDW/5aEktX9bufBKLRRMdU7bNtpLfcGu94B3cdEJgjg==",
906 | "dev": true,
907 | "dependencies": {
908 | "@aashutoshrathi/word-wrap": "^1.2.3",
909 | "deep-is": "^0.1.3",
910 | "fast-levenshtein": "^2.0.6",
911 | "levn": "^0.4.1",
912 | "prelude-ls": "^1.2.1",
913 | "type-check": "^0.4.0"
914 | },
915 | "engines": {
916 | "node": ">= 0.8.0"
917 | }
918 | },
919 | "node_modules/p-limit": {
920 | "version": "3.1.0",
921 | "resolved": "https://registry.npmjs.org/p-limit/-/p-limit-3.1.0.tgz",
922 | "integrity": "sha512-TYOanM3wGwNGsZN2cVTYPArw454xnXj5qmWF1bEoAc4+cU/ol7GVh7odevjp1FNHduHc3KZMcFduxU5Xc6uJRQ==",
923 | "dev": true,
924 | "dependencies": {
925 | "yocto-queue": "^0.1.0"
926 | },
927 | "engines": {
928 | "node": ">=10"
929 | },
930 | "funding": {
931 | "url": "https://github.com/sponsors/sindresorhus"
932 | }
933 | },
934 | "node_modules/p-locate": {
935 | "version": "5.0.0",
936 | "resolved": "https://registry.npmjs.org/p-locate/-/p-locate-5.0.0.tgz",
937 | "integrity": "sha512-LaNjtRWUBY++zB5nE/NwcaoMylSPk+S+ZHNB1TzdbMJMny6dynpAGt7X/tl/QYq3TIeE6nxHppbo2LGymrG5Pw==",
938 | "dev": true,
939 | "dependencies": {
940 | "p-limit": "^3.0.2"
941 | },
942 | "engines": {
943 | "node": ">=10"
944 | },
945 | "funding": {
946 | "url": "https://github.com/sponsors/sindresorhus"
947 | }
948 | },
949 | "node_modules/parent-module": {
950 | "version": "1.0.1",
951 | "resolved": "https://registry.npmjs.org/parent-module/-/parent-module-1.0.1.tgz",
952 | "integrity": "sha512-GQ2EWRpQV8/o+Aw8YqtfZZPfNRWZYkbidE9k5rpl/hC3vtHHBfGm2Ifi6qWV+coDGkrUKZAxE3Lot5kcsRlh+g==",
953 | "dev": true,
954 | "dependencies": {
955 | "callsites": "^3.0.0"
956 | },
957 | "engines": {
958 | "node": ">=6"
959 | }
960 | },
961 | "node_modules/path-exists": {
962 | "version": "4.0.0",
963 | "resolved": "https://registry.npmjs.org/path-exists/-/path-exists-4.0.0.tgz",
964 | "integrity": "sha512-ak9Qy5Q7jYb2Wwcey5Fpvg2KoAc/ZIhLSLOSBmRmygPsGwkVVt0fZa0qrtMz+m6tJTAHfZQ8FnmB4MG4LWy7/w==",
965 | "dev": true,
966 | "engines": {
967 | "node": ">=8"
968 | }
969 | },
970 | "node_modules/path-is-absolute": {
971 | "version": "1.0.1",
972 | "resolved": "https://registry.npmjs.org/path-is-absolute/-/path-is-absolute-1.0.1.tgz",
973 | "integrity": "sha512-AVbw3UJ2e9bq64vSaS9Am0fje1Pa8pbGqTTsmXfaIiMpnr5DlDhfJOuLj9Sf95ZPVDAUerDfEk88MPmPe7UCQg==",
974 | "dev": true,
975 | "engines": {
976 | "node": ">=0.10.0"
977 | }
978 | },
979 | "node_modules/path-key": {
980 | "version": "3.1.1",
981 | "resolved": "https://registry.npmjs.org/path-key/-/path-key-3.1.1.tgz",
982 | "integrity": "sha512-ojmeN0qd+y0jszEtoY48r0Peq5dwMEkIlCOu6Q5f41lfkswXuKtYrhgoTpLnyIcHm24Uhqx+5Tqm2InSwLhE6Q==",
983 | "dev": true,
984 | "engines": {
985 | "node": ">=8"
986 | }
987 | },
988 | "node_modules/prelude-ls": {
989 | "version": "1.2.1",
990 | "resolved": "https://registry.npmjs.org/prelude-ls/-/prelude-ls-1.2.1.tgz",
991 | "integrity": "sha512-vkcDPrRZo1QZLbn5RLGPpg/WmIQ65qoWWhcGKf/b5eplkkarX0m9z8ppCat4mlOqUsWpyNuYgO3VRyrYHSzX5g==",
992 | "dev": true,
993 | "engines": {
994 | "node": ">= 0.8.0"
995 | }
996 | },
997 | "node_modules/punycode": {
998 | "version": "2.3.0",
999 | "resolved": "https://registry.npmjs.org/punycode/-/punycode-2.3.0.tgz",
1000 | "integrity": "sha512-rRV+zQD8tVFys26lAGR9WUuS4iUAngJScM+ZRSKtvl5tKeZ2t5bvdNFdNHBW9FWR4guGHlgmsZ1G7BSm2wTbuA==",
1001 | "dev": true,
1002 | "engines": {
1003 | "node": ">=6"
1004 | }
1005 | },
1006 | "node_modules/queue-microtask": {
1007 | "version": "1.2.3",
1008 | "resolved": "https://registry.npmjs.org/queue-microtask/-/queue-microtask-1.2.3.tgz",
1009 | "integrity": "sha512-NuaNSa6flKT5JaSYQzJok04JzTL1CA6aGhv5rfLW3PgqA+M2ChpZQnAC8h8i4ZFkBS8X5RqkDBHA7r4hej3K9A==",
1010 | "dev": true,
1011 | "funding": [
1012 | {
1013 | "type": "github",
1014 | "url": "https://github.com/sponsors/feross"
1015 | },
1016 | {
1017 | "type": "patreon",
1018 | "url": "https://www.patreon.com/feross"
1019 | },
1020 | {
1021 | "type": "consulting",
1022 | "url": "https://feross.org/support"
1023 | }
1024 | ]
1025 | },
1026 | "node_modules/resolve-from": {
1027 | "version": "4.0.0",
1028 | "resolved": "https://registry.npmjs.org/resolve-from/-/resolve-from-4.0.0.tgz",
1029 | "integrity": "sha512-pb/MYmXstAkysRFx8piNI1tGFNQIFA3vkE3Gq4EuA1dF6gHp/+vgZqsCGJapvy8N3Q+4o7FwvquPJcnZ7RYy4g==",
1030 | "dev": true,
1031 | "engines": {
1032 | "node": ">=4"
1033 | }
1034 | },
1035 | "node_modules/reusify": {
1036 | "version": "1.0.4",
1037 | "resolved": "https://registry.npmjs.org/reusify/-/reusify-1.0.4.tgz",
1038 | "integrity": "sha512-U9nH88a3fc/ekCF1l0/UP1IosiuIjyTh7hBvXVMHYgVcfGvt897Xguj2UOLDeI5BG2m7/uwyaLVT6fbtCwTyzw==",
1039 | "dev": true,
1040 | "engines": {
1041 | "iojs": ">=1.0.0",
1042 | "node": ">=0.10.0"
1043 | }
1044 | },
1045 | "node_modules/rimraf": {
1046 | "version": "3.0.2",
1047 | "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-3.0.2.tgz",
1048 | "integrity": "sha512-JZkJMZkAGFFPP2YqXZXPbMlMBgsxzE8ILs4lMIX/2o0L9UBw9O/Y3o6wFw/i9YLapcUJWwqbi3kdxIPdC62TIA==",
1049 | "dev": true,
1050 | "dependencies": {
1051 | "glob": "^7.1.3"
1052 | },
1053 | "bin": {
1054 | "rimraf": "bin.js"
1055 | },
1056 | "funding": {
1057 | "url": "https://github.com/sponsors/isaacs"
1058 | }
1059 | },
1060 | "node_modules/run-parallel": {
1061 | "version": "1.2.0",
1062 | "resolved": "https://registry.npmjs.org/run-parallel/-/run-parallel-1.2.0.tgz",
1063 | "integrity": "sha512-5l4VyZR86LZ/lDxZTR6jqL8AFE2S0IFLMP26AbjsLVADxHdhB/c0GUsH+y39UfCi3dzz8OlQuPmnaJOMoDHQBA==",
1064 | "dev": true,
1065 | "funding": [
1066 | {
1067 | "type": "github",
1068 | "url": "https://github.com/sponsors/feross"
1069 | },
1070 | {
1071 | "type": "patreon",
1072 | "url": "https://www.patreon.com/feross"
1073 | },
1074 | {
1075 | "type": "consulting",
1076 | "url": "https://feross.org/support"
1077 | }
1078 | ],
1079 | "dependencies": {
1080 | "queue-microtask": "^1.2.2"
1081 | }
1082 | },
1083 | "node_modules/sax": {
1084 | "version": "1.2.4",
1085 | "resolved": "https://registry.npmjs.org/sax/-/sax-1.2.4.tgz",
1086 | "integrity": "sha512-NqVDv9TpANUjFm0N8uM5GxL36UgKi9/atZw+x7YFnQ8ckwFGKrl4xX4yWtrey3UJm5nP1kUbnYgLopqWNSRhWw=="
1087 | },
1088 | "node_modules/shebang-command": {
1089 | "version": "2.0.0",
1090 | "resolved": "https://registry.npmjs.org/shebang-command/-/shebang-command-2.0.0.tgz",
1091 | "integrity": "sha512-kHxr2zZpYtdmrN1qDjrrX/Z1rR1kG8Dx+gkpK1G4eXmvXswmcE1hTWBWYUzlraYw1/yZp6YuDY77YtvbN0dmDA==",
1092 | "dev": true,
1093 | "dependencies": {
1094 | "shebang-regex": "^3.0.0"
1095 | },
1096 | "engines": {
1097 | "node": ">=8"
1098 | }
1099 | },
1100 | "node_modules/shebang-regex": {
1101 | "version": "3.0.0",
1102 | "resolved": "https://registry.npmjs.org/shebang-regex/-/shebang-regex-3.0.0.tgz",
1103 | "integrity": "sha512-7++dFhtcx3353uBaq8DDR4NuxBetBzC7ZQOhmTQInHEd6bSrXdiEyzCvG07Z44UYdLShWUyXt5M/yhz8ekcb1A==",
1104 | "dev": true,
1105 | "engines": {
1106 | "node": ">=8"
1107 | }
1108 | },
1109 | "node_modules/strip-ansi": {
1110 | "version": "6.0.1",
1111 | "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.1.tgz",
1112 | "integrity": "sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A==",
1113 | "dev": true,
1114 | "dependencies": {
1115 | "ansi-regex": "^5.0.1"
1116 | },
1117 | "engines": {
1118 | "node": ">=8"
1119 | }
1120 | },
1121 | "node_modules/strip-json-comments": {
1122 | "version": "3.1.1",
1123 | "resolved": "https://registry.npmjs.org/strip-json-comments/-/strip-json-comments-3.1.1.tgz",
1124 | "integrity": "sha512-6fPc+R4ihwqP6N/aIv2f1gMH8lOVtWQHoqC4yK6oSDVVocumAsfCqjkXnqiYMhmMwS/mEHLp7Vehlt3ql6lEig==",
1125 | "dev": true,
1126 | "engines": {
1127 | "node": ">=8"
1128 | },
1129 | "funding": {
1130 | "url": "https://github.com/sponsors/sindresorhus"
1131 | }
1132 | },
1133 | "node_modules/supports-color": {
1134 | "version": "7.2.0",
1135 | "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz",
1136 | "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==",
1137 | "dev": true,
1138 | "dependencies": {
1139 | "has-flag": "^4.0.0"
1140 | },
1141 | "engines": {
1142 | "node": ">=8"
1143 | }
1144 | },
1145 | "node_modules/text-table": {
1146 | "version": "0.2.0",
1147 | "resolved": "https://registry.npmjs.org/text-table/-/text-table-0.2.0.tgz",
1148 | "integrity": "sha512-N+8UisAXDGk8PFXP4HAzVR9nbfmVJ3zYLAWiTIoqC5v5isinhr+r5uaO8+7r3BMfuNIufIsA7RdpVgacC2cSpw==",
1149 | "dev": true
1150 | },
1151 | "node_modules/type-check": {
1152 | "version": "0.4.0",
1153 | "resolved": "https://registry.npmjs.org/type-check/-/type-check-0.4.0.tgz",
1154 | "integrity": "sha512-XleUoc9uwGXqjWwXaUTZAmzMcFZ5858QA2vvx1Ur5xIcixXIP+8LnFDgRplU30us6teqdlskFfu+ae4K79Ooew==",
1155 | "dev": true,
1156 | "dependencies": {
1157 | "prelude-ls": "^1.2.1"
1158 | },
1159 | "engines": {
1160 | "node": ">= 0.8.0"
1161 | }
1162 | },
1163 | "node_modules/type-fest": {
1164 | "version": "0.20.2",
1165 | "resolved": "https://registry.npmjs.org/type-fest/-/type-fest-0.20.2.tgz",
1166 | "integrity": "sha512-Ne+eE4r0/iWnpAxD852z3A+N0Bt5RN//NjJwRd2VFHEmrywxf5vsZlh4R6lixl6B+wz/8d+maTSAkN1FIkI3LQ==",
1167 | "dev": true,
1168 | "engines": {
1169 | "node": ">=10"
1170 | },
1171 | "funding": {
1172 | "url": "https://github.com/sponsors/sindresorhus"
1173 | }
1174 | },
1175 | "node_modules/uri-js": {
1176 | "version": "4.4.1",
1177 | "resolved": "https://registry.npmjs.org/uri-js/-/uri-js-4.4.1.tgz",
1178 | "integrity": "sha512-7rKUyy33Q1yc98pQ1DAmLtwX109F7TIfWlW1Ydo8Wl1ii1SeHieeh0HHfPeL2fMXK6z0s8ecKs9frCuLJvndBg==",
1179 | "dev": true,
1180 | "dependencies": {
1181 | "punycode": "^2.1.0"
1182 | }
1183 | },
1184 | "node_modules/web-streams-polyfill": {
1185 | "version": "3.2.1",
1186 | "resolved": "https://registry.npmjs.org/web-streams-polyfill/-/web-streams-polyfill-3.2.1.tgz",
1187 | "integrity": "sha512-e0MO3wdXWKrLbL0DgGnUV7WHVuw9OUvL4hjgnPkIeEvESk74gAITi5G606JtZPp39cd8HA9VQzCIvA49LpPN5Q==",
1188 | "engines": {
1189 | "node": ">= 8"
1190 | }
1191 | },
1192 | "node_modules/which": {
1193 | "version": "2.0.2",
1194 | "resolved": "https://registry.npmjs.org/which/-/which-2.0.2.tgz",
1195 | "integrity": "sha512-BLI3Tl1TW3Pvl70l3yq3Y64i+awpwXqsGBYWkkqMtnbXgrMD+yj7rhW0kuEDxzJaYXGjEW5ogapKNMEKNMjibA==",
1196 | "dev": true,
1197 | "dependencies": {
1198 | "isexe": "^2.0.0"
1199 | },
1200 | "bin": {
1201 | "node-which": "bin/node-which"
1202 | },
1203 | "engines": {
1204 | "node": ">= 8"
1205 | }
1206 | },
1207 | "node_modules/wrappy": {
1208 | "version": "1.0.2",
1209 | "resolved": "https://registry.npmjs.org/wrappy/-/wrappy-1.0.2.tgz",
1210 | "integrity": "sha512-l4Sp/DRseor9wL6EvV2+TuQn63dMkPjZ/sp9XkghTEbV9KlPS1xUsZ3u7/IQO4wxtcFB4bgpQPRcR3QCvezPcQ==",
1211 | "dev": true
1212 | },
1213 | "node_modules/xml2js": {
1214 | "version": "0.6.2",
1215 | "resolved": "https://registry.npmjs.org/xml2js/-/xml2js-0.6.2.tgz",
1216 | "integrity": "sha512-T4rieHaC1EXcES0Kxxj4JWgaUQHDk+qwHcYOCFHfiwKz7tOVPLq7Hjq9dM1WCMhylqMEfP7hMcOIChvotiZegA==",
1217 | "dependencies": {
1218 | "sax": ">=0.6.0",
1219 | "xmlbuilder": "~11.0.0"
1220 | },
1221 | "engines": {
1222 | "node": ">=4.0.0"
1223 | }
1224 | },
1225 | "node_modules/xmlbuilder": {
1226 | "version": "11.0.1",
1227 | "resolved": "https://registry.npmjs.org/xmlbuilder/-/xmlbuilder-11.0.1.tgz",
1228 | "integrity": "sha512-fDlsI/kFEx7gLvbecc0/ohLG50fugQp8ryHzMTuW9vSa1GJ0XYWKnhsUx7oie3G98+r56aTQIUB4kht42R3JvA==",
1229 | "engines": {
1230 | "node": ">=4.0"
1231 | }
1232 | },
1233 | "node_modules/yocto-queue": {
1234 | "version": "0.1.0",
1235 | "resolved": "https://registry.npmjs.org/yocto-queue/-/yocto-queue-0.1.0.tgz",
1236 | "integrity": "sha512-rVksvsnNCdJ/ohGc6xgPwyN8eheCxsiLM8mxuE/t/mOVqJewPuO1miLpTHQiRgTKCLexL4MeAFVagts7HmNZ2Q==",
1237 | "dev": true,
1238 | "engines": {
1239 | "node": ">=10"
1240 | },
1241 | "funding": {
1242 | "url": "https://github.com/sponsors/sindresorhus"
1243 | }
1244 | }
1245 | }
1246 | }
1247 |
--------------------------------------------------------------------------------
/package.json:
--------------------------------------------------------------------------------
1 | {
2 | "type": "module",
3 | "name": "db-tickets",
4 | "version": "1.1.0",
5 | "description": "A library to retrieve ticket information from Deutsche Bahn",
6 | "main": "index.js",
7 | "scripts": {
8 | "test": "node test/index.js",
9 | "lint": "./node_modules/.bin/eslint --fix --ext .js lib",
10 | "prepublishOnly": "npm run lint && npm test"
11 | },
12 | "author": "envake",
13 | "contributors": [
14 | "krmax44 ",
15 | "arnebr (https://github.com/arnebr)",
16 | "Jannis R "
17 | ],
18 | "license": "ISC",
19 | "dependencies": {
20 | "node-fetch": "^3.2.8",
21 | "xml2js": "^0.6.2"
22 | },
23 | "devDependencies": {
24 | "eslint": "^8.20.0"
25 | },
26 | "engines": {
27 | "node": ">=16"
28 | }
29 | }
30 |
--------------------------------------------------------------------------------
/test/index.js:
--------------------------------------------------------------------------------
1 | import { fileURLToPath } from 'node:url'
2 | import { readFileSync } from 'node:fs'
3 | import { deepStrictEqual } from 'node:assert';
4 | import { parseTicket } from '../lib/queryTicket.js'
5 |
6 | const PATH_456800002504 = 'ticket-456800002504-2023-07-10-erika-musterfrau.json'
7 | const TICKET_456800002504 = readFileSync(
8 | fileURLToPath(new URL(PATH_456800002504, import.meta.url).href),
9 | { encoding: 'utf8' },
10 | );
11 | const PATH_XXXXXX = 'ticket-XXXXXX-2023-05-12-erika-musterman.json'
12 | const TICKET_XXXXXX = readFileSync(
13 | fileURLToPath(new URL(PATH_XXXXXX, import.meta.url).href),
14 | { encoding: 'utf8' },
15 | );
16 |
17 | {
18 | const ticket = JSON.parse(TICKET_456800002504)
19 |
20 | const parsed = parseTicket(ticket)
21 | deepStrictEqual(parsed, {
22 | order: {
23 | ticketNumber: '456800002504',
24 | bookingDate: '2023-07-10T17:01:51',
25 | validFrom: '2023-07-23T00:00:00',
26 | validUntil: '2023-07-24T10:00:00',
27 | journeyStart: '2023-07-23T00:00:00',
28 | firstName: 'Erika',
29 | lastName: 'Musterfrau',
30 | text: 'Super Sparpreis, 2. Kl., Büchen/Berlin Hbf (tief)',
31 | class: 'S2'
32 | },
33 | journey: {
34 | type: 'journey',
35 | id: undefined, // todo: fix this
36 | legs: [
37 | {
38 | origin: {
39 | type: 'station',
40 | id: '8000058',
41 | name: 'Büchen',
42 | location: { type: 'location' },
43 | },
44 | destination: {
45 | type: 'station',
46 | id: '8098160',
47 | name: 'Berlin Hbf (tief)',
48 | location: { type: 'location' },
49 | },
50 | line: {
51 | type: 'line',
52 | id: '1',
53 | name: 'ICE 509',
54 | mode: 'train',
55 | },
56 | departure: '2023-07-23T10:45:00+00:00',
57 | arrival: '2023-07-23T12:20:00+00:00',
58 | arrivalPlatform: '1',
59 | mode: 'train',
60 | public: true,
61 | }
62 | ],
63 | price: {
64 | amount: '35.90',
65 | currency: 'EUR',
66 | },
67 | },
68 | })
69 |
70 | console.info('ok parseTicket works with 456800002504')
71 | const ticket_xxxxx = JSON.parse(TICKET_XXXXXX)
72 |
73 | const parsed_xxxxx = parseTicket(ticket_xxxxx)
74 | deepStrictEqual(parsed_xxxxx, {
75 | order: {
76 | ticketNumber: 'XXXXXX',
77 | bookingDate: '2023-05-12T23:57:51',
78 | validFrom: '2023-05-14T00:00:00',
79 | validUntil: '2023-05-15T23:59:59',
80 | journeyStart: '2023-05-14T13:38:00',
81 | firstName: 'Erika',
82 | lastName: 'Musterman',
83 | text: 'Einfache Fahrt, Flexpreis, 2. Kl., 1 Erw., BC 50, Berlin+City/Hamburg+City#',
84 | class: 'S2'
85 | },
86 | journey: {
87 | type: 'journey',
88 | id: '12922018',
89 | legs: [
90 | {
91 | origin: {
92 | type: 'station',
93 | id: '8098160',
94 | name: 'Berlin Hbf (tief)',
95 | location: {
96 | latitude: 52.525589,
97 | longitude: 13.369549,
98 | type: 'location'
99 | },
100 | },
101 | destination: {
102 | type: 'station',
103 | id: '8002553',
104 | name: 'Hamburg-Altona',
105 | location: {
106 | latitude: 53.552697,
107 | longitude: 99.35175,
108 | type: 'location'
109 | },
110 | },
111 | line: {
112 | type: 'line',
113 | id: 'C0-0.0',
114 | name: 'ICE 1600',
115 | mode: 'train'
116 | },
117 | departure: '2023-05-14T13:38:00+00:00',
118 | departurePlatform: '8',
119 | arrival: '2023-05-14T15:39:00+00:00',
120 | arrivalPlatform: '6',
121 | mode: 'train',
122 | public: true
123 | }
124 | ],
125 | price: { amount: '42.90', currency: 'EUR' }
126 | }
127 | }
128 | )
129 |
130 | console.info('ok parseTicket works with XXXXXX')
131 | }
132 |
--------------------------------------------------------------------------------
/test/ticket-456800002504-2023-07-10-erika-musterfrau.json:
--------------------------------------------------------------------------------
1 | {
2 | "rporderdetails": {
3 | "$": {
4 | "version": "2.0"
5 | },
6 | "rpheader": [
7 | {
8 | "$": {
9 | "tnr": "",
10 | "ts": "2023-09-11T18:40:35"
11 | }
12 | }
13 | ],
14 | "order": [
15 | {
16 | "$": {
17 | "cdt": "2023-07-10T17:01:51",
18 | "ddt": "2024-07-23T09:00:00+02:00",
19 | "fkat": "5",
20 | "kwid": "ced723bf-48f2-4706-8a7d-000000ed402d",
21 | "ldt": "2023-07-10T17:01:51",
22 | "on": "456800002504",
23 | "pg": "2",
24 | "sdt": "2023-07-23T00:00:00",
25 | "version": "17",
26 | "vfrom": "2023-07-23T00:00:00",
27 | "vto": "2023-07-24T10:00:00",
28 | "zweg": "99"
29 | },
30 | "txt": [
31 | "Büchen - Berlin Hbf (tief)"
32 | ],
33 | "schedulelist": [
34 | {
35 | "out": [
36 | {
37 | "$": {
38 | "posnr": "1"
39 | },
40 | "txt": [
41 | "Büchen 10:45 - Berlin Hbf (tief) 12:20"
42 | ],
43 | "trainlist": [
44 | {
45 | "train": [
46 | {
47 | "$": {
48 | "tid": "1",
49 | "tn": "ICE 509",
50 | "type": "T"
51 | },
52 | "zugnr": [
53 | "509"
54 | ],
55 | "gat": [
56 | "ICE"
57 | ],
58 | "dep": [
59 | {
60 | "$": {
61 | "dt": "2023-07-23T00:00:00",
62 | "t": "10:45:00"
63 | },
64 | "n": [
65 | "Büchen"
66 | ],
67 | "nr": [
68 | "8000058"
69 | ]
70 | }
71 | ],
72 | "arr": [
73 | {
74 | "$": {
75 | "dt": "2023-07-23T00:00:00",
76 | "t": "12:20:00"
77 | },
78 | "n": [
79 | "Berlin Hbf (tief)"
80 | ],
81 | "nr": [
82 | "8098160"
83 | ],
84 | "ptf": [
85 | "1"
86 | ]
87 | }
88 | ]
89 | }
90 | ]
91 | }
92 | ]
93 | }
94 | ]
95 | }
96 | ],
97 | "tcklist": [
98 | {
99 | "tck": [
100 | {
101 | "$": {
102 | "posnr": "2"
103 | },
104 | "htdata": [
105 | {
106 | "ht": [
107 | {
108 | "_": "PCFET0NUWVBFIGh0bWwKICBQVUJMSUMgIi0vL1czQy8vRFREIEhUTUwgNC4wIFRyYW5zaXRpb25hbC8vRU4iPgo8aHRtbCB4bWxucz0iaHR0cDovL3d3dy53My5vcmcvMTk5OS94aHRtbCI+PGhlYWQ+PG1ldGEgaHR0cC1lcXVpdj0iQ29udGVudC1UeXBlIiBjb250ZW50PSJ0ZXh0L2h0bWw7IGNoYXJzZXQ9dXRmLTgiPjwvbWV0YT48bWV0YSBuYW1lPSJ2aWV3cG9ydCIgY29udGVudD0id2lkdGg9ZGV2aWNlLXdpZHRoLCBpbml0aWFsLXNjYWxlPTEuMCwgbWluaW11bS1zY2FsZT0xLjAsIHVzZXItc2NhbGFibGU9bm8iPjwvbWV0YT48c3R5bGUgeG1sbnM9IiIgdHlwZT0idGV4dC9jc3MiPgogICAgYm9keSB7CiAgICAgICAgd29yZC1icmVhazogYnJlYWstd29yZDsKICAgICAgICBtYXJnaW46IDA7CiAgICAgICAgcGFkZGluZzogMDsKICAgIH0KCiAgICBpbWcgewogICAgICAgIC13ZWJraXQtdXNlci1zZWxlY3Q6IG5vbmU7CiAgICAgICAgLXdlYmtpdC10b3VjaC1jYWxsb3V0OiBub25lOwogICAgfQoKICAgIC5pbmxpbmUtdGV4dCB7CiAgICAgICAgd2hpdGUtc3BhY2U6IHByZS13cmFwOwogICAgICAgIGRpc3BsYXk6IGlubGluZS1ibG9jazsKICAgIH0KCiAgICAuaW5saW5lLXRleHQtZG9udC1icmVhayB7CiAgICB9CgogICAgLmJsb2NrLWNvbnRhaW5lciB7CiAgICAgICAgd2hpdGUtc3BhY2U6IHByZS13cmFwOwogICAgICAgIG1heC13aWR0aDogMTAwJTsKICAgIH0KCiAgICAuYmxvY2sgewogICAgICAgIG1heC13aWR0aDogMTAwJTsKICAgIH0KCiAgICAjdGlja2V0YmFyY29kZSB7CiAgICAgICAgd2lkdGg6IDEwMCU7CiAgICAgICAgaW1hZ2UtcmVuZGVyaW5nOiBjcmlzcC1lZGdlczsKICAgICAgICBpbWFnZS1yZW5kZXJpbmc6IHBpeGVsYXRlZDsKICAgIH0KCiAgICAjYm90dG9tTG9nb0hhbmR5ewogICAgICAgIHdpZHRoOiAxMDAlOwogICAgICAgIGhlaWdodDogMTVtbQogICAgfQoKICAgICNib3R0b21Mb2dvVG91cmlzdGlrewogICAgICAgIHdpZHRoOiAxMDAlOwogICAgICAgIGhlaWdodDogN21tCiAgICB9CgogICAgI3NpY2h0cHJ1ZWZtZXJrbWFsIHsKICAgICAgICB3aWR0aDogMTAwJTsKICAgIH0KCiAgICAjbm9BbmltYXRpb25Mb2dvIHsKICAgICAgICBkaXNwbGF5Om5vbmU7CiAgICB9CgogICAgI2FuaW1hdGlvbkxvZ28gewogICAgICAgIHBvc2l0aW9uOiByZWxhdGl2ZTsKICAgICAgICBhbmltYXRpb246IG1vdmVMb2dvIDMwcyBpbmZpbml0ZTsKICAgICAgICBhbmltYXRpb24tdGltaW5nLWZ1bmN0aW9uOiBsaW5lYXI7CiAgICB9CgogICAgQGtleWZyYW1lcyBtb3ZlTG9nbyB7CiAgICAgICAgZnJvbSB7bGVmdDogLTI1bW07fQogICAgICAgIHRvIHtsZWZ0OiAxMDAlO30KICAgIH0KPC9zdHlsZT48c2NyaXB0IHhtbG5zPSIiPjwvc2NyaXB0Pjx0aXRsZT5EZXV0c2NoZSBCYWhuIEhhbmR5dGlja2V0PC90aXRsZT48L2hlYWQ+PGJvZHk+PGRpdiBzdHlsZT0iZm9udC1mYW1pbHk6IHNhbnMtc2VyaWY7IGZvbnQtc2l6ZTogMS4wZW07Ij48ZGl2IHZhbGlnbj0idG9wIj48YSBuYW1lPSJCYXJjb2RlIj48L2E+PGRpdj48aW1nIHNyYz0iZGF0YTppbWFnZS9wbmc7YmFzZTY0LGlWQk9SdzBLR2dvQUFBQU5TVWhFVWdBQUFWd0FBQUZjQVFBQUFBQ3NiVHVCQUFBQ3ZVbEVRVlI0QWRYWXAyR0FSeGpHNEpjWmFnVHZ2MFUyOENwaWdVcm5sLzVaK0NGL3VicFNDOGxWQ05TR0RGbzJXTFJET0EwWERHSk13TlZFM1RUSk0zZ0dBbFVrbTRIWkdNQmk2bllHNjNRV1pMbVZWWlZyak5iU0xkMlBQZlF2NHcxZWJMdUQ3U254RGg0K2ZaUnhCa084NEsvMENxNzFrcllyR0pBWDNMeUM1K3liWWZYcDUxZnY0T2xITDNFSFI2K1R6Q0hNeXdPYVovQ0twd2ZrRXQ0K25qQmV3Zm4yZ0s0ejJBVTlCR2R3K1lhM1hjRTgycHBYY0kyM245OWR3Ulk4V08wTUp1eWhyK0VWYkw1OWxEeUVnWmNIbEozQjVndFd2WU8vMy80NTllTUpzeXZZclJmOFZWN0JHVDMwdzlnVjdQUUZPNzJDMi9PUGRBYjdpSXRkd2EzZUxwM3lDcFo0d2RIK1pheXVWanAvanhtYkRaeUx0ckFZRFdEU3FHUk1hbU5HYzJld3BMWU1GbXUvMTFwdU9xWDBDZzZqTlpjNFhGWFNBZ3BYUTNjRkErUmlwaEM2YlJaT0tGTmRaN0NNMFZZVm9sWXFGQnNOeE9ZVnZDWEQ1UnhqNnFaazJ3YmJxanlEWGF3R2k1RFVWUGNyakkwUXVZUGZPNFBIc0tRU0dXb3FFU1prR3QzQndwWUdEWnU2YWRXTUt0ZTZneHZab0NSTHJkUllrekpXdURQWWFvVGxiT2kyaWN5TmtUTHFEdVozUHhtU3E4b3hHbTRiTGZRT0pzUmFxdUIralpwSXJhRnhCNnZGeU1tZ0ZvSkJnZ05XdWlzWTFFR05Wc01sTmN4Vk5RZm9GYnpSUTBxN2d1a0paN3VEdDk2Nmc5cytYeVNIOE9pbFZYZHdiM2lIc0srOFBJUDFDYXFIOEY3d3JGM0IyVXZLdklLSkY0eTFLM2pzOHdXM085aDR3c2daek9pdG5jSGhpMlRnRmJ6eCtDTjVCb3M5SkxJcnVKN3diR2V3Ni9NSmIxN0JzQmRzc1N2WXJZY1lPNE0zbno1S20xZHc5SUpwbmNIZmNmOWNmZmJFMngzTUN4M2pES1k5WVQyRTdmTUpkd2YzMWxLdllNZWJwbDNCK0xab1RzN2c2UXVPUElOZFQxYjVDNWdLVjVoTi9oQlVhZmtYY0Z0dFk3YmxCcHRNVWZyZGIzVUdLeXVYQ3pReElsQ01MWEp6Vi9CUGNrUDg4TFVwemUwQUFBQUFTVVZPUks1Q1lJST0iIGFsdD0iVGlja2V0IEJhcmNvZGUiIGlkPSJ0aWNrZXRiYXJjb2RlIj48L2ltZz48L2Rpdj48ZGl2IHN0eWxlPSJib3JkZXItdG9wLWNvbG9yOiBibGFjazsgYm9yZGVyLXRvcC1zdHlsZTogc29saWQ7IGJvcmRlci10b3Atd2lkdGg6IDAuMm1tOyBtYXJnaW4tdG9wOiAxLjBleDsiPjxzcGFuPiZuYnNwOzwvc3Bhbj48c3Bhbj48L3NwYW4+PC9kaXY+PGEgbmFtZT0iVGlja2V0aW5oYWJlciI+PC9hPjxkaXY+PHNwYW4+RXJpa2E8L3NwYW4+PHNwYW4+IDwvc3Bhbj48c3Bhbj5NdXN0ZXJmcmF1PC9zcGFuPjwvZGl2PjxkaXYgc3R5bGU9ImZvbnQtd2VpZ2h0OiA3MDA7IG1hcmdpbi10b3A6IDIuMG1tOyI+PHNwYW4+Q0lWIDEwODA8L3NwYW4+PC9kaXY+PGRpdiBzdHlsZT0ibWFyZ2luLXRvcDogMS4wZXg7Ij48L2Rpdj48YSBuYW1lPSJHdWVsdGlna2VpdCI+PC9hPjxkaXYgc3R5bGU9Im1hcmdpbi10b3A6IDEuMGV4OyI+PGEgbmFtZT0iR3VlbHRpZ2tlaXRIZWFkZXIiPjwvYT48ZGl2IHN0eWxlPSJmb250LXNpemU6IDEyLjBwdDsgZm9udC13ZWlnaHQ6IDcwMDsgdGV4dC1kZWNvcmF0aW9uOiBuby11bmRlcmxpbmU7Ij48c3Bhbj5Hw7xsdGlna2VpdDwvc3Bhbj48L2Rpdj48ZGl2PjxzcGFuPklDRTwvc3Bhbj48c3Bhbj4gPC9zcGFuPjxzcGFuPkZhaHJrYXJ0ZTwvc3Bhbj48c3Bhbj4gKDwvc3Bhbj48c3Bhbj5FaW5mYWNoZSBGYWhydDwvc3Bhbj48c3Bhbj4pPC9zcGFuPjwvZGl2PjxkaXY+PHNwYW4+U3VwZXIgU3BhcnByZWlzPC9zcGFuPjwvZGl2PjxkaXY+PHNwYW4+PC9zcGFuPjwvZGl2PjxkaXY+PHNwYW4+Mi4gS2xhc3NlPC9zcGFuPjwvZGl2PjxkaXY+PHNwYW4+MTwvc3Bhbj48c3Bhbj4gPC9zcGFuPjxzcGFuPlBlcnNvbiAoMjctNjQgSmFocmUpPC9zcGFuPjxzcGFuPiBtaXQgPC9zcGFuPjxzcGFuPjEgQkMyNTwvc3Bhbj48L2Rpdj48ZGl2PjxzcGFuPlZvbjogPC9zcGFuPjxzcGFuPjIzLjA3LjIwMjMsIDAwOjAwIFVocjwvc3Bhbj48L2Rpdj48ZGl2PjxzcGFuPkJpczogPC9zcGFuPjxzcGFuPjI0LjA3LjIwMjMsIDEwOjAwIFVocjwvc3Bhbj48L2Rpdj48L2Rpdj48YSBuYW1lPSJWZXJiaW5kdW5nMiI+PC9hPjxkaXYgc3R5bGU9Im1hcmdpbi10b3A6IDEuMGV4OyI+PGRpdiBzdHlsZT0iZm9udC1zaXplOiAxMi4wcHQ7IGZvbnQtd2VpZ2h0OiA3MDA7IHRleHQtZGVjb3JhdGlvbjogbm8tdW5kZXJsaW5lOyI+PHNwYW4+VmVyYmluZHVuZzwvc3Bhbj48L2Rpdj48ZGl2PjxzcGFuPkLDvGNoZW48L3NwYW4+PHNwYW4+IC0gPC9zcGFuPjxzcGFuPkJlcmxpbjwvc3Bhbj48L2Rpdj48ZGl2PjxzcGFuPlp1Z2JpbmR1bmc6PC9zcGFuPjwvZGl2PjxkaXY+PHNwYW4+SUNFIDUwOTwvc3Bhbj48c3Bhbj4sICA8L3NwYW4+PHNwYW4+MTA6NDUgVWhyPC9zcGFuPjwvZGl2PjxkaXY+PHNwYW4+VmlhOiAmbHQ7MTA4MCZndDsoTFdMKk5FVUQqTkFVL1VFKlNBVypTREwqV1VNKSpCU1A8L3NwYW4+PC9kaXY+PC9kaXY+PGEgbmFtZT0iQnVjaHVuZ3NkZXRhaWxzIj48L2E+PGRpdiBzdHlsZT0ibWFyZ2luLXRvcDogMS4wZXg7Ij48YSBuYW1lPSJCdWNodW5nc2RldGFpbHNIZWFkZXIiPjwvYT48ZGl2IHN0eWxlPSJmb250LXNpemU6IDEyLjBwdDsgZm9udC13ZWlnaHQ6IDcwMDsgdGV4dC1kZWNvcmF0aW9uOiBuby11bmRlcmxpbmU7Ij48c3Bhbj5CdWNodW5nc2RldGFpbHM8L3NwYW4+PC9kaXY+PGRpdj48c3Bhbj5HZWJ1Y2h0IGFtOiA8L3NwYW4+PHNwYW4+MTAuMDcuMjAyMyB1bSAxNzowMSBVaHI8L3NwYW4+PC9kaXY+PGRpdj48c3Bhbj5BdWZ0cmFncy1OcjogPC9zcGFuPjxzcGFuPjQ1NjgwMDAwMjUwNDwvc3Bhbj48L2Rpdj48ZGl2PjxzcGFuPkdlc2FtdHByZWlzOiA8L3NwYW4+PHNwYW4+MzUsOTA8L3NwYW4+PHNwYW4+IDwvc3Bhbj48c3Bhbj7igqw8L3NwYW4+PC9kaXY+PC9kaXY+PGRpdiBzdHlsZT0ibWFyZ2luLXRvcDogMS4wZXg7Ij48ZGl2IHN0eWxlPSJmb250LXNpemU6IDEyLjBwdDsgZm9udC13ZWlnaHQ6IDcwMDsgdGV4dC1kZWNvcmF0aW9uOiBuby11bmRlcmxpbmU7Ij48c3Bhbj5Lb25kaXRpb25lbjwvc3Bhbj48L2Rpdj48ZGl2PjxkaXY+PHNwYW4+WnVnYmluZHVuZzogR2lsdCBudXIgZsO8ciBlaW5nZXRyYWdlbmUgWsO8Z2U8L3NwYW4+PC9kaXY+PC9kaXY+PGRpdiBzdHlsZT0ibWFyZ2luLXRvcDogMS4wZXg7Ij48ZGl2PjxzcGFuPk51ciBnw7xsdGlnIG1pdCBhbXRsaWNoZW4gTGljaHRiaWxkYXVzd2Vpcy4gRGllc2VyIGlzdCBiZWkgZGVyIEtvbnRyb2xsZSB2b3J6dXplaWdlbjwvc3Bhbj48L2Rpdj48ZGl2PjxzcGFuPkJlaSBGYWhya2FydGVuIG1pdCBCYWhuQ2FyZC1SYWJhdHQgemVpZ2VuIFNpZSBiaXR0ZSB6dXPDpHR6bGljaCBJaHJlIGfDvGx0aWdlIEJhaG5DYXJkIHZvcjwvc3Bhbj48L2Rpdj48ZGl2PjxzcGFuPkVzIGdlbHRlbiBkaWUgQmVmw7ZyZGVydW5nc2JlZGluZ3VuZ2VuIGRlciBEQiBBRzwvc3Bhbj48L2Rpdj48ZGl2PjxzcGFuPkJlaSBlaW5lciB6dSBlcndhcnRlbmRlbiBWZXJzcMOkdHVuZyBhYiAyMCBNaW51dGVuIGFtIFppZWxiYWhuaG9mIElocmVyIEZhaHJrYXJ0ZSBpc3QgZGllIFp1Z2JpbmR1bmcgSWhyZXIgRmFocnQgb2huZSBiZXNvbmRlcmUgQmVzY2hlaW5pZ3VuZyBhdWZnZWhvYmVuPC9zcGFuPjwvZGl2PjwvZGl2PjxkaXYgc3R5bGU9Im1hcmdpbi10b3A6IDEuMGV4OyI+PGRpdj48c3Bhbj5TdG9ybmllcnVuZyBhdXNnZXNjaGxvc3Nlbjwvc3Bhbj48L2Rpdj48L2Rpdj48L2Rpdj48ZGl2IHN0eWxlPSJtYXJnaW4tdG9wOiAxLjBleDsiPjxkaXYgc3R5bGU9ImZvbnQtc2l6ZTogMTAuMHB0OyB0ZXh0LWFsaWduOiBsZWZ0OyI+PHNwYW4+VGlja2V0Y29kZTogPC9zcGFuPjxzcGFuPjgxTDdXVzZaPC9zcGFuPjwvZGl2PjwvZGl2PjxkaXYgc3R5bGU9ImJvcmRlci1ib3R0b20tY29sb3I6IGJsYWNrOyBib3JkZXItYm90dG9tLXN0eWxlOiBzb2xpZDsgYm9yZGVyLWJvdHRvbS13aWR0aDogMC4xbW07IG1hcmdpbi10b3A6IDEuMGV4OyI+PHNwYW4+Jm5ic3A7PC9zcGFuPjwvZGl2PjxkaXY+PGltZyBzcmM9ImRhdGE6aW1hZ2UvcG5nO2Jhc2U2NCxpVkJPUncwS0dnb0FBQUFOU1VoRVVnQUFBVXdBQUFFV0NBQUFBQUFvRVBFU0FBQVJ6a2xFUVZSNEFleloxMS9xVGg3RzhmM1R0L2ZlTDdmZGJPOTd0ZVZKQ0NJMm1tQVhrRU1Ua0Y0bGVmYndIWkl3RkgvaVVmQ0M5NDBWbWRjbnlUaVRmSTZ2WnVmVll1N3NZdTVpN21MdVl0YlRyaDQvelM2bTg5UFB1Njc0YVhZeG81L2Z4WHl0bU9QdjdXSytXc3k5eis5aXZsYk13VGRlTSthZ1ArVDJqUHZEdFY4eGZzMlkvL244WXN4NlpzR0lNeDR1RGcwQW9YaTY3dEExeXArR01CRk9sV3h1bVAxd2s1QjNONDZ1bXZRVU12Tnk5RGlsdUFYQWloY2RMbWhscHB3MVlyYS9zaVJtR2dzNjlOUU80TnNyVURocEU3NWduaHZrVkpNbVprUzduRHJDdkgyNmFpRzRRbFc2Rmw0NVhpUG1QejYvWmt6bkNwb1VKeDVQb0VzNTNKZ2M1cGpWejQ1NWgxbDMxSlh3Z3BqVkwwNGFmdkVMejQ3cHBMQXNaZ3p6cnJjWUUyYmpzMklXb05NdkpXZi9KVEYvSnczLzlwVm54N3pCc3BobExHcHZNU1pDOXRNeDJ5WjBwamJjRDNoQnpMd2svRXJyMlRFZnNEVG1NYWFDQnhhbXpyY1pFN21uWTBZd0wwTGZPUGlTbUwrVWhQL2hjMk02WVNpaGRMM2ZyOTlGVmN3aEZLdEtPZ1ZqMm5Xek1jMUVybHk2dGpCMThHVE1HcFJRdG53WGdsS2pKNDBYeEx5V2d0OGNMSThaVzF3YUZhR2NqYW4wenM1bXhuYlBpVnNvd3czR0RLVFZBRWZlTmRMM1kxNHVMSTFpMDdKRGtxTjlpQmhkdzhCTFl2NWNDb2E0UEdhV0N3NGhUdWtiKzQyREZJTk5UNW81SklkZWlDQ1VpaGR6Y1NCRFkrWjNXSUV3aHB5NmxOWHltakVURXZENzQyZkhiRUZZSStvS0VGRXFGa1NURzFJc1Uyalh4WWVWTWIzeEJoMU9PTlArQlNvOVNWMWVMNmI5QXdrWTQ3TmpwbGVzZXU0aDRsVDJJTHJjaGdhVTdCTXhVeEJKS2ttSTFPeFBvNC9yeFF4THY1L3orVEdQVjF5L0xZZ2pDc2ZBaEdGekd3WlFjay9FREh1OVJSWWlUTkhHUkd1OW1NTnZTYi9yMVRFN3BXem1RM1ZFbDYwcVdaem5XQ3FmK3RVNnhBbTNvZytsNU1kc05ZcVpUS0ZoYzJvTTVaN0t2ZFl0cGs3YTlXSkM4djJTSzJNYVVDSVZLbTJJR0VlWll4UFlUOVU0ZFFGeElja2pFSGx1UlIxSzI0OXBRSmlwRGtVSCtxVGVuSDNKZzd5aXQxN003bGNsWDJGRlRFMXNPSHNFei9NQnVKMzdGQU1MSWxHcGw0OGhEaHh1eFEyRTVhaVlHdU5Hdmx1RjBxUFNnMUwxcHJKTHJoZnpYMUx2RDN4T1RPd1BabllhRm56QkZrWE5oRzZ2eTYydzkveUxSSThwVHAyWjFmS1F5aEJLMFQxaEFzUDFZdGEvTkluM3BZZm54Y1NSN1IxMFhYQkEwUXhENktmczVtV2dkUFNZbm1zNUtmUk9ZeWc1ZDR1WDVub3gveWp4L3NWbnhwUy9mNGtsWWxTY0MzaU1ITGVrWVVDY2MwVk1OTWtzRkp1SzdhK204bktDak5lTFdaU2JibC90UEJrekdJQW44RWllWXhsMWNyZFBNQ3M1NERZTTlpQkNqMXJNa0g3d00xQWNLZzZVRE1meStqelhpL2tiYVFldWpCbjUwSGJJWVg1djVvYmZHYWFNOUlDdEJHYjJsdmNMYzJhYm16YzZnRERxZEdPYXlYS2ZaT2ZhOERmdG1SVm5aa2I5Sk95c0dWUHFmWHU0SW1iMnFzK3A0YUYvUk04eGxaSGplUUpoMm1UVGdISWNQWFJyRHJocDlvay8rU21Kbk5laWJucjd6THNWYytiZEtPQXRQOWVPK2VPL3VMNzRlZVZYa3k5Nm5OVTFJQUwrbkduWjJyM05PcDBEaUpNK3lmWStSSW9iNXNTZnZKUDZBVXFTZVNpUHBOWXRmeTNuQTllUHVWcURtaVNVQWRQZWhTMmNBRVNlRllpOVJ5MS9uNXQxQmlYbWNCazdDSEhJTXBUQjNBNjBiTXE1OFpZeEMxQmEzckc5MVI5WHBabUN5T2o1YzF0WnJlTjR6T1ZPSVlLc2U4c25mVWRVTnlZL3ppaTNVRzQvZmo1NHRaZzE3MnF1UUs4VWc3am1nYjQ5eTIvak9zOUJDWSs0d2hXRXliNjJEdkhucTc2QjVacXZGdlBCZStzTzlGMTNBdUtHaHI2anFHN2hUa2NKU3FqUFZhN2RtSTRKVWRTdlBkTjUrNWdsN3k4NjV0SXpNMnRER2VuNUQ3ZzVOUVBDNm5DbEZJUkZua0JjNjVXUCtmWXhML3lIS2hGOUVBY1FKUWRLVDcrbGRjaU5hUWE4NStXcmhkeGh1ZlVpRkY3Yk40bDVXYUZ2RlBDV1JzeG9POGl4QWRGbVFMOC9lT3VOZFVPNlFRaWpTazIwUzE4RlN0TDdOMkNNdENkQ3RaZkd6S1kxWC82OFlrMitHQ2NSNzNES1NjQXYySVV3MVNES0VKYWpQMmZ6bmdlZmNVTUdJU2dsNmt6ajVuRitwNGtQL2xPZm05bXJQT2lzRlhNMWZRZVVsRHVUc2pKdlJUQ1ZkeThJTjlPajMrd2NTbm54a2RZR2VHOW83TSs0bk1RRUFsZFN3aW03TGRIMyt4bEZra1hEbTd4S00vSlFDaDgvSDMxaVRNQTRpa1pEY0ZuajJVc2xYcWtYdy81aHEyRHFyRnEvVDNpajNuQk0zYW5FbkxCT29wRUFYQWwrNUMyQ3d0RXdGRU1mcnI1by8rU1l1anVLeU5KQk8vdFlGT2Y3aUtrejJ0NEtYM2ZEamNXTU9CUzl3Tktidy9kWVlIYmVaY3dzaFgwQTNZRzlzWmhISTA1VkRjd0tOUDNkaGE3STl4anpnbE85b0g1TzlQaEdNYXRIMENYSDlOUXMrRUl0VG1YbUlwZjVEbUptOXFBeGMvVDB3dkNGdTN5VG1LSnhzUWRQN0lHekJtZUdXK3htVEU4bmFmb3BMd2Q4RHpGcGwwNU5QK1c1ZHZyWmFRdUtkV3R6blpqcjY1U3Z6NU94ODV2eWtQT0d4YXRVOVB5MmFsTXpybWJQVDZQSjg3c0htKytIM2N4Zm5zY1Q1OW1hdmZDanlzMTU5T3ltWXYrZmZUc3BBQUNFQVJpR2YxVTRBeEdjMjFJSitmZlZpS3JXdFMyWU1HSENGRXlZTUw4TUprekJoQWtUcG1EQ2hDbVlNR0hDRkV5WU1BVVRKa3lZZ2dsenhDZ0tKa3lZTUNjNzl3RGx1QnBIQVR5RHRURnMxN1p0MjdadDI3WnQyN1p0MjdzMVJuV1QreWI1MHZlMXl6bDZTSHVQb3Y4OXA3OWFjY1N3WG9DNUwxOFFTZWFQSU9uU1NJRTJ2VFg0V2FabHpmU0c0eGU0eG1HdFZEOVZzM0Q0SWRaK2lXc3F2QUJ6YTlveXJkcng2ZXF5NmRWU2lkby9sOEtJaE15NkdIN2hRUWFteGs5NVZxYjZ5V2tPYnFhcmM5VHNGWmg3ZnJ5WXY4Wk1rN2FGc0dkVzVpUzE0NDY1TStpVURWNkdlZUZNOUpXVnhpdm5UQVR6eFk0YlZsaHZ6WnAxMXNpNU1GTjNsei9uQUxaaXAvU3htS2VPeHdBNGVjb0VQSjgzWU00REd5NTBpRC8rb2dXNHNXRHE4U2hPckR3NUtQRzRzMmJTVGd2cHNDUXgyOVp0RUwvSWw4Ymx2d3FZOTNLVmY4dCtLY25rek9NZmRNTWhZaWJibjNGVk5IQTMvZmE4dFJTb1hlVWJnSnAxRmRnZEhGSThNR0NUYVZBSUV6SkNieDZhTUdsR3YwcXZXVkxaV3M0RWQxU1RkbHBJaHlXRTJXdjFSajVQN1dpYklNOHFKU3RpM3NsUjZSMkhHU0hIcmJnb0cyTVFNUk8vcWRGRUNjekkrcmFZTzZhamNEMGx0QzByZmlWMzgyVXBwNnU1bTluYXFNWEtuVUZYSGVJaUxaUWtKa095S0FwdEEvaUhOb0o1TUh1MWp4eHdkbjgwWU03ZFJTdGlKbnl6S093cDZ5elhWZU9CYWN4UitCUHcrYkpKd0l3dVVPd0xnT0ZCenpsU1NURDVSVm9vVGN6MVgzVjh6QnphNXYwRUVUTkhBbVorSkRsdjA1S0JCZjNjTUo5bldoWjlPLzB1c3djbUJpY0k2cjVMeFVMQWZKYXB5cXdsUzVhMFRYRFJUaW9KSnI5SUMrbXdSQjh6eTN4eFlmclZMNXoycmhNNFZjSS9UOHRsSVc2WXFOTlFPVFg3TzFETWNyVVZjSnpwbUlyeEgya1FNQy9JNU1WSzhybGlJNVVFazEra2hYUlk4cGdWdjl6TDBFQUpZNzdRQjA1WWMzZDF3MXdlZXE5a0R5M0JyQnpyRVptemh1RGh1RkUzNEtTTngzeWVxWi9PbzVKaTBrSTZMSG5NVmlwTVNMd3U1bkhHWVhyZ3F0ejlsdmttYzZORWV5MENacGVjSDRHejRUVVVWM1B1TXdPbnd6ZWJlRXhUa1Z5ZkFRekw4LzRIVEZwSWg2V0UyV1RNUkNGZldVOU1ZOG5RcC9wOFdZNi8zNU0zb01FNzltOU1OUFFQL2dBQmMyR0tvZGQzNVVwU1cyRXNXdXFxNlVXalZQZWQrME1tUHJOdlNqUG8yYWY1S2NjYmY3eGwwa0xYc0JTZnpaazdUazlNSEFwcHB6bFp4bzlKZjJSYThsWnFpcms2ZFE4dHdZeHFGOGhrVzFxL29RS1hDc1EycE5sbWdxcHhRQmV0YzNVUXd5U2VaTUFQbUtDRmROaHJQalhpbm4xd0FBbzlpNTlHOTg1R0ZreVBUdHcwQ2dkcFRCd1FkZnVLbHYxOUlSMldCdWEvSHgrbUQ5T0g2Y3YvR3ZNdmR1aUNBQUFZZ0FGUS85UjdocnRCQkdRaVU2Wk1tY2lVZVI2Wk1tVWlVNlpNbWNpVUtST1pNbVVpVTZaTW1jaVVLUk9aTW1YS1JLWk1tY2lVS1ZNbU1tWEtSS1pNbVRLUktWTW1NbVhLUktaTW1US1JLVk1tTW1YS2xJbE1tVEtSS1ZPbVRHVEtsSWxNbVRLUktWT21UR1RLbElsTW1USmxJbE9tVEdUS2xDa1RtVEpsSWxPbVRKbklsQ2tUbVRKbElsT21USm5JbENrVG1USmx5a1NtVEpuSWxDbFRKakpsWm9uVm1XR1lsc3pTdm5tb1NhdnlBUGorciszL1QrKzlnMTBSeFFia1BLc2JKSzdPR2Jjd1g1bjNxVTZOTHlGQVpuZFNvaXp5VWlnRFcwd3JpcXlzT3dzRTI4bXl5SXBTOWhiZUhlelFWbVdlNzBWbE8xRm1wZWpzNjhyODlQOExBaGFHS21LT3ZBY2ZYZUlUWEhoaGpBVm5DSzgwQktSbU94U3pMdVZIVlU1RVpjM3hpWHBIcDJFTDlvVXl1M1FibUFWSHo5bEtQT0xEa2hHNGVoZGsxdEVtcWdZY1U4SldrdW5OWkJac1MrTG1lcmNKYjRDWmltMVI3NEJNenJaVXppWDFIRTNoWkxJTVk5akdGMm1xMk1ISGQxRW1rekJqRTBaSmJEaVpHRVMyRzdTTjJWUHlkMUltVzhaWXNDM2liV1VtV2JaSzR0YlB3THpyVzN4dThCS3o3UG91eHd2OWpzamthWlp4T3NZYXIrSXNjZUcrbmN4ODJUQ01MaFhiaDh2MDBkbjhiYkc3S0xGT2tudHJnc3VNUGFwRkpoZkRzcXhIdmpUaDNaUmFVMU41dFBnU3BaUzVMRlBXQkpUSkh5NXN3ZklSa053YjBZbk5jQU93UnNFQkVreUdHZXNzQjBMSEI0c2VaOEo1MEpFM3hwSHp0NXFOQU5nQncyV1phRytQb2VyOGRPZk9HVWl5R2xsWGhUaTlseHBmRmdxM05IZEF5Q1pZNmRiNTM3TUZReVowL3lZeWdWQ3QweU9qdFRwQmgyeGhvbUVuRUl5QnU1M2FNYkVMUzVEUnh1SWxBc2hVN2hNdHg0bENLb0MwYnFBWCt1QXlYYTVGSXh4UzRuU21TZUhjcGdGazltNFdURmlUcVV3Qkc1bGQwR2xPaHp6V2NJUnc5WXJUKzJqeGlYQXlCNVMwelV5QjgzMmd3UlVRa29ZZTFpN0oxQzQ5NkIzcXQ1Zlo0anJqNHExcHJhbWhJQnQ3dkd3Z0pPaUtwUmIyS1hDYUQ1c2FQNkd6K1dtNVVPUE5TaW4xcThrczNBSlVNNXB5T0Zzd1pTTkRZdE1RbHZMeTBjdEdXTW9WRGRCbHF1S3owbk9iZHBUNTlXL0lKNDhlZjN5NG9CSlFEVi9YZFc0Mzh4L25lYW9CWUl4UmVXQnN6aGJLaTNXZ2NIUE5BUFhXaXZrbW5pZnpDM1BRejVUQ0NkWEpPdDRsT2FqYmRGVTc0RE5wbHBBT1NEQXdJbHBnOHNhNUdMaDdVbTQ4V2J5dFpVK3R6c3I4YVhiSERtVFdqR1hOK0tDeWpWQlQ2OGxrdFlZeDl5ZTk0b3dTalJBZWsyeFdqSlR4UW1rQW1BUkQ5TEhNcFpERloyV3FlV0ovT2g3Sm5JazhROXlnVEFxdWlXUEtmQW9OZ1NGbnh0aVFKVEttelN5VUNZaVR1ZFRWOXFSTStIV1c5K2NGbVpSNnB3Tk15ajE1dG9FYk1kQ3ltZTV1YzQ0emN5bW5rVDBwYzVoVDg1UCtTcG1KZmRJRXBLVnhpSmxQcHVFMm9LanhRS2JBcFdoSFpvTkg1UHFrVFBoenR2ZnJkVEpqN2EySDBVNXA3UGxXOGdTQm9SdWdha2NtRnZqRDFSeFBiOXljbERrdEF0dHJaR2JHUHlhV0xkOG1vT2F1YWF4UzNNNEZoelRVZG1VS3ZJL2RUWHZuM2xLZGxBbHMxdmZEZjh2TU9sZ1lzVW9hbVhMRzRxTDM2eVdtZzgxdVdqY0hmNTRUbWJ5Y3FMeGg3eTM5NHZta1RQTUYranVRS1NjbFpkTVpRS3czNkJUdXo1c2VtekRCSWNlWlZlWTR0RksyZzlOaU43MWtSYnpsY3prNEtSUGkyZCszeHpKaFMrcEdrREpTZlJqR1RXVTJuc3lEenFha2kxYnMzOHg0VXFiOWFoYVlYUyt6ZHBXSG91Z3VLY0tTZE10cExpL0lMSEUxSXQySDByL01Uc3FFZkJiNGxiMWE1dWdhR3BTV05qQTVscVJBdElvTytIRm0wbmdqT3ZndFdZL1VTWm53N1d3d3ZsWW0rZVVNc1RRNGI2S040VnFaaFJ0ZVEzNE9PcENKRVhaa2NkZGtPWTNQeWhTendjLzF2c3hjZWhoU3FrdUxvMWlXWHYrb0k3bGh3c2xrdkRhTHk1UTVOVTVtSlZjYXY2MGRHKzhuNjN6ekcrSlptZkREclBDZlhabVVpZjVWU1Z6M1dnOHlXeW9OMXBlbzg3YkVVZGdtTzg4YnBVVGtqbXdISjZDWWRBOWpxU1NtWUU5THhYbVo3YXp3MC9FcW1SZ0ZwU1JCUndsM1R3U1Z1YVc1S0JNeXRpWGJIS1BPeTRSZlpvZS9YeTBUeEs1TXhaNHkzbEptYkMvTEhQbkZ2ek5ybmlXelcvb2RuMXd0RTRvOW1aRHZkQlJ1S0pNUGNGa210THVaN0loUHlVVCtXQ3hlTHhPcVBaa20zVHEyTjVUSk96aVdTWGJxaUFTS2VwYk04ZE96TXFGUDJFclVZdEhtL3FNTmhLUXJPQ2wvRTF5UWlmVHgrbUFIVzVKVE1wRi9kbVFPOGdrR1BQb3FtV1BJNng0Y3BpbVc4SkpDV1FpTTdVWCsrTzFpQUVjcnQvaWpyUExvWWVEemRpL0o4QTF3UnFiK25NaDhJVm9idUIxV205UHZzSy82cnl2Umtjdzc1MlhhTDE5TjVsMG1aSzhuOHk0VHZuazltWGVaUTQxb09NOWQ1cDI3ekx2TXU4dzdkNWtCK0JjZmZEWCtRcGpwS1FBQUFBQkpSVTVFcmtKZ2dnPT0iIGFsdD0iU2ljaHRwcnVlZm1lcmttYWwiIGlkPSJzaWNodHBydWVmbWVya21hbCI+PC9pbWc+PC9kaXY+PGEgbmFtZT0iX2xhc3QtYmxvY2siPjwvYT48ZGl2PjwvZGl2PjwvZGl2PjwvZGl2PjwvYm9keT48L2h0bWw+",
109 | "$": {
110 | "name": "httext",
111 | "pos": "1",
112 | "type": "text/html;base64"
113 | }
114 | }
115 | ]
116 | }
117 | ],
118 | "mtk": [
119 | {
120 | "$": {
121 | "dir": "1",
122 | "status": "8"
123 | },
124 | "txt": [
125 | "Super Sparpreis, 2. Kl., Büchen/Berlin Hbf (tief)"
126 | ],
127 | "zb": [
128 | "N"
129 | ],
130 | "iss": [
131 | "80"
132 | ],
133 | "reisender_vorname": [
134 | "Erika"
135 | ],
136 | "reisender_nachname": [
137 | "Musterfrau"
138 | ],
139 | "nvplist": [
140 | {
141 | "nvp": [
142 | {
143 | "_": "-------",
144 | "$": {
145 | "name": "k_erm_db"
146 | }
147 | },
148 | {
149 | "_": "1",
150 | "$": {
151 | "name": "anzerw"
152 | }
153 | },
154 | {
155 | "_": "0",
156 | "$": {
157 | "name": "anzkind"
158 | }
159 | },
160 | {
161 | "_": "S2",
162 | "$": {
163 | "name": "klasse"
164 | }
165 | },
166 | {
167 | "_": "8",
168 | "$": {
169 | "name": "status"
170 | }
171 | }
172 | ]
173 | }
174 | ]
175 | }
176 | ]
177 | }
178 | ]
179 | }
180 | ],
181 | "posinfolist": [
182 | {
183 | "posinfo": [
184 | {
185 | "$": {
186 | "dir": "1",
187 | "dirlabel": "none",
188 | "posnr": "2",
189 | "shownr": "1002",
190 | "state": "8",
191 | "type": "TCK",
192 | "typeinfo": "0",
193 | "vfo": "2023-07-23T00:00:00",
194 | "vto": "2023-07-24T10:00:00"
195 | },
196 | "childlist": [
197 | {
198 | "posinfo": [
199 | {
200 | "$": {
201 | "dir": "1",
202 | "dirlabel": "none",
203 | "posnr": "1",
204 | "shownr": "1100",
205 | "state": "8",
206 | "type": "EVA",
207 | "typeinfo": "0",
208 | "vfo": "2023-07-23T10:45:00",
209 | "vto": "2023-07-23T12:20:00"
210 | },
211 | "childlist": [
212 | ""
213 | ]
214 | }
215 | ]
216 | }
217 | ]
218 | }
219 | ]
220 | }
221 | ]
222 | }
223 | ]
224 | }
225 | }
226 |
--------------------------------------------------------------------------------
/test/ticket-XXXXXX-2023-05-12-erika-musterman.json:
--------------------------------------------------------------------------------
1 | {
2 | "rporderdetails": {
3 | "$": {
4 | "version": "2.0"
5 | },
6 | "rpheader": [
7 | {
8 | "$": {
9 | "tnr": "",
10 | "ts": "2023-09-12T08:38:02"
11 | }
12 | }
13 | ],
14 | "order": [
15 | {
16 | "$": {
17 | "cdt": "2023-05-12T23:57:51",
18 | "cid": "12922018",
19 | "ddt": "2024-07-12T23:57:51",
20 | "fkat": "5",
21 | "hkey": "dc6cc67efa0f0019ed8e305525ba28f2",
22 | "ldt": "2023-05-12T23:57:51",
23 | "on": "XXXXXX",
24 | "pg": "1",
25 | "sdt": "2023-05-14T13:38:00",
26 | "version": "18",
27 | "vfrom": "2023-05-14T00:00:00",
28 | "vto": "2023-05-15T23:59:59",
29 | "zweg": "1"
30 | },
31 | "schedulelist": [
32 | {
33 | "out": [
34 | {
35 | "$": {
36 | "posnr": "1"
37 | },
38 | "txt": [
39 | "Berlin Hbf (tief) 13:38 - Hamburg-Altona 15:39"
40 | ],
41 | "tp": [
42 | {
43 | "depn": [
44 | "Berlin+City"
45 | ],
46 | "arrn": [
47 | "Hamburg+City"
48 | ]
49 | }
50 | ],
51 | "trainlist": [
52 | {
53 | "train": [
54 | {
55 | "$": {
56 | "tid": "C0-0.0",
57 | "tn": "ICE 1600",
58 | "type": "T"
59 | },
60 | "zugnr": [
61 | "1600"
62 | ],
63 | "gat": [
64 | "ICE"
65 | ],
66 | "sci": [
67 | "Y"
68 | ],
69 | "dep": [
70 | {
71 | "$": {
72 | "dt": "2023-05-14T00:00:00",
73 | "t": "13:38:00"
74 | },
75 | "n": [
76 | "Berlin Hbf (tief)"
77 | ],
78 | "nr": [
79 | "8098160"
80 | ],
81 | "ebhf_name": [
82 | "Berlin Hbf (tief)"
83 | ],
84 | "ebhf_nr": [
85 | "8098160"
86 | ],
87 | "ptf": [
88 | "8"
89 | ],
90 | "plz": [
91 | "10557"
92 | ],
93 | "x": [
94 | "13369549"
95 | ],
96 | "y": [
97 | "52525589"
98 | ]
99 | }
100 | ],
101 | "arr": [
102 | {
103 | "$": {
104 | "dt": "2023-05-14T00:00:00",
105 | "t": "15:39:00"
106 | },
107 | "n": [
108 | "Hamburg-Altona"
109 | ],
110 | "nr": [
111 | "8002553"
112 | ],
113 | "ebhf_name": [
114 | "Hamburg-Altona"
115 | ],
116 | "ebhf_nr": [
117 | "8002553"
118 | ],
119 | "ptf": [
120 | "6"
121 | ],
122 | "plz": [
123 | "22765"
124 | ],
125 | "x": [
126 | "9935175"
127 | ],
128 | "y": [
129 | "53552697"
130 | ]
131 | }
132 | ]
133 | }
134 | ]
135 | }
136 | ]
137 | }
138 | ]
139 | }
140 | ],
141 | "tcklist": [
142 | {
143 | "tck": [
144 | {
145 | "$": {
146 | "posnr": "2"
147 | },
148 | "htdata": [
149 | {
150 | "ht": [
151 | {
152 | "_": "",
153 | "$": {
154 | "name": "barcode",
155 | "pos": "1",
156 | "type": "image/png;base64"
157 | }
158 | },
159 | {
160 | "_": "PHA+RnJhdcKgIEVyaWthIE11c3Rlcm1hbjxici8+Cjxici8+CklDRSBGYWhya2FydGUsIEZsZXhwcmVpcyAoRWluZmFjaGUgRmFocnQpPGJyLz4KRmFocnRhbnRyaXR0IGFtIDE0LjA1LjIwMjM8YnIvPgpDaXR5IGFtIDE0LjA1LjIzPGJyLz4KPGJyLz4KS2xhc3NlOiAyLCBFcncuOiAxLCBtaXQgMcKgQkM1MDxici8+CkhpbmZhaHJ0OiBCZXJsaW4rQ2l0eSAtIEhhbWJ1cmcrQ2l0eSwgbWl0IElDRTxici8+Cjxici8+CsOcYmVyOiBWSUE6IEJTUCooV1VNKlNETCpTQVcqVUUqSEFSL1dCRSpMV0wpPGJyLz4KPGJyLz4KTnVyIGfDvGx0aWcgbWl0IGFtdGxpY2hlbSBMaWNodGJpbGRhdXN3ZWlzICh6LkIuIFBlcnNvbmFsYXVzd2VpcykuIERpZXNlciBpc3QgYmVpIGRlciBLb250cm9sbGUgdm9yenV6ZWlnZW4uPGJyLz4KQmVpIEZhaHJrYXJ0ZW4gbWl0IEJhaG5DYXJkLVJhYmF0dCB6ZWlnZW4gU2llIGJpdHRlIHp1c8OkdHpsaWNoIElocmUgZ8O8bHRpZ2UgQmFobkNhcmQgdm9yLjxici8+Cjxici8+ClN0b3JubyBrb3N0ZW5mcmVpIGJpcyAxIFRhZyB2b3IgMS4gR2VsdHVuZ3N0YWcuPGJyLz4KPGJyLz4KQXVmdHJhZ3MtTnI6IFhYWFhYWDxici8+Ckdlc2FtdHByZWlzOiA0Miw5MCBFVVI8YnIvPgpHZWJ1Y2h0IGFtIDEyLjA1LjIwMjMgdW0gMjM6NTcgVWhyPGJyLz4KPC9wPg==",
161 | "$": {
162 | "name": "httext",
163 | "pos": "2",
164 | "type": "text/html;base64"
165 | }
166 | },
167 | {
168 | "_": "\n\t\tbody {\n\t\t\tfont-family: sans-serif;\n\t\t\tfont-size: 16px;\n\t\t\tfont-size: 1rem;\n\t\t\ttext-align: center;\n\t\t}\n\n\t\t.barcode img,\n\t\t.sichtmerkmal img {\n\t\t\twidth: 90%;\n\t\t}\n\n\t\tp.AngebotBezeichnung,\n\t\t.httext {\n\t\t\tpadding-left: 1em;\n\t\t\tpadding-right: 1em;\n\t\t\ttext-align: left;\n\t\t\tline-height: 1.1;\n\t\t\tword-break: break-word;\n\t\t}\n\n\t\t.httext h4 {\n\t\t\tfont-weight: bold;\n\t\t\tfont-size: 1.1rem;\n\t\t}\n\n\t\tp.CityMobilTarifangaben {\n\t\t\tmargin: 0 1em;\n\t\t\tpadding-left: 0;\n\t\t\tpadding-right: 0;\n\t\t\tfont-weight: bold;\n\t\t}\n\t\t\n\t\t.ShtAngebotbezeichnung {\n\t\t\tfont-weight: bold;\n\t\t\tfont-size: 1.1rem;\n\t\t}\n\t\t\n\t\t.OsaIPSIProduktbezeichnung {\n\t\t\tfont-weight: bold;\n\t\t\tfont-size: 1.1rem;\n\t\t}\n\t\t\n\t\t.OsaIPSIKondition {\n\t\t\tfont-size: 0.8rem;\n\t\t}\n\t",
169 | "$": {
170 | "name": "htstyle",
171 | "pos": "2",
172 | "type": "text/css"
173 | }
174 | },
175 | {
176 | "_": "",
177 | "$": {
178 | "name": "sichtmerkmal",
179 | "pos": "10",
180 | "type": "image/png;base64"
181 | }
182 | }
183 | ]
184 | }
185 | ],
186 | "mtk": [
187 | {
188 | "$": {
189 | "dir": "1",
190 | "status": "8"
191 | },
192 | "txt": [
193 | "Einfache Fahrt, Flexpreis, 2. Kl., 1 Erw., BC 50, Berlin+City/Hamburg+City#"
194 | ],
195 | "zb": [
196 | "N"
197 | ],
198 | "iss": [
199 | "80"
200 | ],
201 | "tkey": [
202 | "XXXXXX-5"
203 | ],
204 | "ot_nr_hin": [
205 | "800000048"
206 | ],
207 | "reisender_vorname": [
208 | "Erika"
209 | ],
210 | "reisender_nachname": [
211 | "Musterman"
212 | ],
213 | "bc": [
214 | {
215 | "rbs": [
216 | "50"
217 | ],
218 | "anz": [
219 | "1"
220 | ]
221 | }
222 | ],
223 | "nvplist": [
224 | {
225 | "nvp": [
226 | {
227 | "_": "1 BC 50",
228 | "$": {
229 | "name": "k_erm_db"
230 | }
231 | },
232 | {
233 | "_": "1",
234 | "$": {
235 | "name": "anzerw"
236 | }
237 | },
238 | {
239 | "_": "0",
240 | "$": {
241 | "name": "anzkind"
242 | }
243 | },
244 | {
245 | "_": "S2",
246 | "$": {
247 | "name": "klasse"
248 | }
249 | },
250 | {
251 | "_": "NP",
252 | "$": {
253 | "name": "preisartcodedb"
254 | }
255 | },
256 | {
257 | "_": "Berlin Hbf (tief)",
258 | "$": {
259 | "name": "tpevahstart"
260 | }
261 | },
262 | {
263 | "_": "Hamburg-Altona",
264 | "$": {
265 | "name": "tpevahziel"
266 | }
267 | },
268 | {
269 | "_": "8",
270 | "$": {
271 | "name": "status"
272 | }
273 | },
274 | {
275 | "_": "Berlin+City",
276 | "$": {
277 | "name": "citystart"
278 | }
279 | },
280 | {
281 | "_": "Hamburg+City",
282 | "$": {
283 | "name": "cityziel"
284 | }
285 | }
286 | ]
287 | }
288 | ]
289 | }
290 | ]
291 | }
292 | ]
293 | }
294 | ],
295 | "posinfolist": [
296 | {
297 | "posinfo": [
298 | {
299 | "$": {
300 | "dir": "1",
301 | "dirlabel": "none",
302 | "posnr": "2",
303 | "shownr": "1002",
304 | "state": "8",
305 | "type": "TCK",
306 | "typeinfo": "0",
307 | "vfo": "2023-05-14T00:00:00",
308 | "vto": "2023-05-15T23:59:59"
309 | },
310 | "childlist": [
311 | {
312 | "posinfo": [
313 | {
314 | "$": {
315 | "dir": "1",
316 | "dirlabel": "none",
317 | "posnr": "1",
318 | "shownr": "1100",
319 | "state": "8",
320 | "type": "EVA",
321 | "typeinfo": "0",
322 | "vfo": "2023-05-14T13:38:00",
323 | "vto": "2023-05-14T16:22:00"
324 | },
325 | "childlist": [
326 | ""
327 | ]
328 | }
329 | ]
330 | }
331 | ]
332 | }
333 | ]
334 | }
335 | ],
336 | "txt": [
337 | "Berlin Hbf (tief) - Hamburg-Altona"
338 | ]
339 | }
340 | ]
341 | }
342 | }
--------------------------------------------------------------------------------